2004-01-15 20:49:41 +01:00
|
|
|
;; libgcc routines for the Renesas H8/300 CPU.
|
2000-09-08 00:11:31 +02:00
|
|
|
;; Contributed by Steve Chamberlain <sac@cygnus.com>
|
2004-01-15 20:49:41 +01:00
|
|
|
;; Optimizations by Toshiyasu Morita <toshiyasu.morita@renesas.com>
|
1994-07-16 22:45:33 +02:00
|
|
|
|
2014-01-02 23:25:22 +01:00
|
|
|
/* Copyright (C) 1994-2014 Free Software Foundation, Inc.
|
1994-07-16 22:45:33 +02:00
|
|
|
|
|
|
|
This file is free software; you can redistribute it and/or modify it
|
|
|
|
under the terms of the GNU General Public License as published by the
|
2009-04-09 17:00:19 +02:00
|
|
|
Free Software Foundation; either version 3, or (at your option) any
|
1994-07-16 22:45:33 +02:00
|
|
|
later version.
|
|
|
|
|
|
|
|
This file 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.
|
|
|
|
|
2009-04-09 17:00:19 +02:00
|
|
|
Under Section 7 of GPL version 3, you are granted additional
|
|
|
|
permissions described in the GCC Runtime Library Exception, version
|
|
|
|
3.1, as published by the Free Software Foundation.
|
|
|
|
|
|
|
|
You should have received a copy of the GNU General Public License and
|
|
|
|
a copy of the GCC Runtime Library Exception along with this program;
|
|
|
|
see the files COPYING3 and COPYING.RUNTIME respectively. If not, see
|
|
|
|
<http://www.gnu.org/licenses/>. */
|
1994-07-16 22:45:33 +02:00
|
|
|
|
1994-07-12 00:45:37 +02:00
|
|
|
/* Assembler register definitions. */
|
|
|
|
|
|
|
|
#define A0 r0
|
|
|
|
#define A0L r0l
|
|
|
|
#define A0H r0h
|
|
|
|
|
|
|
|
#define A1 r1
|
|
|
|
#define A1L r1l
|
|
|
|
#define A1H r1h
|
|
|
|
|
|
|
|
#define A2 r2
|
|
|
|
#define A2L r2l
|
|
|
|
#define A2H r2h
|
|
|
|
|
|
|
|
#define A3 r3
|
|
|
|
#define A3L r3l
|
|
|
|
#define A3H r3h
|
|
|
|
|
|
|
|
#define S0 r4
|
|
|
|
#define S0L r4l
|
|
|
|
#define S0H r4h
|
|
|
|
|
|
|
|
#define S1 r5
|
|
|
|
#define S1L r5l
|
|
|
|
#define S1H r5h
|
|
|
|
|
|
|
|
#define S2 r6
|
|
|
|
#define S2L r6l
|
|
|
|
#define S2H r6h
|
|
|
|
|
|
|
|
#ifdef __H8300__
|
|
|
|
#define PUSHP push
|
|
|
|
#define POPP pop
|
|
|
|
|
|
|
|
#define A0P r0
|
|
|
|
#define A1P r1
|
|
|
|
#define A2P r2
|
|
|
|
#define A3P r3
|
|
|
|
#define S0P r4
|
|
|
|
#define S1P r5
|
|
|
|
#define S2P r6
|
|
|
|
#endif
|
|
|
|
|
Introduce H8SX support.
* expr.c (expand_strcpy): Renamed and moved to...
* builtins.c (expand_movstr): ... here. Tweak.
(expand_builtin_strcpy): Adjust. Use movstr if len can't be
computed or has side effects.
(expand_builtin_stpcpy): Likewise. Use strcpy if return value is
unused, or if mempcpy fails. Adjust the return value in the
latter case. Use movstr if everything else fails.
* doc/md.texi (movstr): Document.
(movmemM, clrmemM): Fix explanation of memory block operands.
* config/h8300/h8300.md (stpcpy): Renamed to...
(movstr): ... this. Adjust.
2004-07-07 Alexandre Oliva <aoliva@redhat.com>
* config/h8300/h8300.md: Rename movstr*, except for movstrict*, to
movmem* and clrstr* to clrmem*.
2004-06-27 Alexandre Oliva <aoliva@redhat.com>
* config/h8300/h8300.c (h8300_reg_class_from_letter): Map 'D' to
GENERAL_REGS, always.
(h8300_swap_into_er6, h8300_swap_into_er6): Handle the case of
getting the stack pointer as addr.
* config/h8300/h8300.h (PREDICATE_CODES): Remove constant rtxes
from general_operand_dst.
* config/h8300/h8300.md (movmd_internal_normal): New, normal-mode
variant of...
(movmd_internal): ... this. Add modes to operands. Disparage `D'
instead of requiring it to match only before reload.
(stpcpy_internal_normal): New, normal-mode variant of...
(stpcpy_internal): ... this. Add modes to operands. Disparage
`D' instead of requiring it to match only before reload.
* config/h8300/h8300-protos.h (h8300_legitimate_address_p): Add
mode argument.
* config/h8300/h8300.h (GO_IF_LEGITIMATE_ADDRESS): Pass it to...
* config/h8300/h8300.c (h8300_legitimate_address_p): Pass it to
h8300_get_index.
* config/h8300/h8300.md (attr type): Add call.
(attr can_delay): If type is call, set it no.
(call, call_value): Set type to call.
2004-06-21 Alexandre Oliva <aoliva@redhat.com>
* config/h8300/h8300.md (logicalhi3_sn, logicalsi3_sn): New.
2004-06-16 Alexandre Oliva <aoliva@redhat.com>
* tree.c (get_narrower): Don't narrow integral types into
non-integral types.
* config/h8300/h8300.c (h8300_expand_epilogue): Initialize
frame_size *before* the first use.
* config/h8300/h8300.md (movstrictqi): Reintroduce post-increment
on input.
(peephole2): Don't widen instructions that push SP. Move
decrement of SP to the end of all stm-generating peepholes.
2003-07-24 Richard Sandiford <rsandifo@redhat.com>
* config/h8300/h8300.md (insv): Prefer to use AND to clear a bitfield
and OR to set it to all ones.
2003-07-24 Richard Sandiford <rsandifo@redhat.com>
* config/h8300/h8300.md (can_delay): Default to "no" for bit branches.
(call, call_value): Set can_delay to "no".
2003-07-22 Richard Sandiford <rsandifo@redhat.com>
* config/h8300/h8300.md (extzv): Make subreg check more robust.
2003-07-21 Richard Sandiford <rsandifo@redhat.com>
* config/h8300/h8300.md (*brabit): Remove.
* config/h8300/h8300.md (*brabc, *brabs): Remove mode from
zero_extract. Use bit_memory_operand as the predicate for
operand 1 and 'WU' as the constraint. Check the difference
between the base length and the final one when deciding which
type of branch to use.
2003-07-21 Richard Sandiford <rsandifo@redhat.com>
* config/h8300/h8300.md (extzv): Remove mode from operands 0 and 1.
Use convert_move to extend the result for TARGET_H8300SX. Check
for QImode memory references. Optimize the case where the
destination is a paradoxical subreg.
2003-07-21 Richard Sandiford <rsandifo@redhat.com>
* config/h8300/h8300.md (*movsf_h8sx): Add an r <- G alternative.
* config/h8300/h8300.md (andqi): Remove bclr from h8sx version.
2003-07-21 Richard Sandiford <rsandifo@redhat.com>
* config/h8300/h8300.md: Include mova.md
(length_table): Add mova and mova_zero.
* config/h8300/h8300.c (print_operand): Handle '%o'. Print a length
after all constant addresses for '%R', '%X', '%T' and '%S'.
(h8300_mova_length): New function.
(h8300_insn_length_from_table): Use it to handle mova and mova_zero.
* config/h8300/t-h8300 (mova.md): Generate from genmova.sh. Add to
dependencies for s-config, etc.
* config/h8300/gemova.sh: New file.
* config/h8300/mova.md: Generated.
2003-07-20 Alexandre Oliva <aoliva@redhat.com>
* config/h8300/h8300.c (h8300_bitfield_length): New.
(nibble_operand): Adjust.
(h8300_binary_length): Handle conditional binary op.
(h8300_insn_length_from_table): Handle bitfield and bitbranch.
* config/h8300/h8300.h: Change constraints W# and Y# to P#>X and
P#<X, respectively. The original P is now IP4>X. Introduced P#>0
and P#<0, unused so far. W and Y are now prefixes to multi-letter
constraints. WU is introduced as a variant of U that requires a
mem, and is therefore considered an EXTRA_MEMORY_CONSTRAINT.
* config/h8300/h8300.md (attr type): Added bitbranch.
(attr length_table): Added bitfield and bitbranch.
(attr length): Compute bitbranch length.
(andqi): Separate pattern for H8300SX. Use bfld for loading the
least-significant bit of a byte.
(brabit, brabc, brabs): New.
(insv, extzv): Emit bfst and bfld on H8300SX.
(bfld, bfst, seq, sne): New.
(bstzhireg, cmpstz, bstz, bistz): New.
(cmpcondbset, condbset, cmpcondbclr, condbclr): New.
(cmpcondbsetreg, condbsetreg, cmpcondbclrreg, condbclrreg): New.
2003-07-11 Richard Sandiford <rsandifo@redhat.com>
* config/h8300/h8300.c (h8sx_binary_memory_operator): New function.
(h8sx_unary_memory_operator): New function.
* config/h8300/h8300.h (EXTRA_MEMORY_CONSTRAINT): Disable.
(PREDICATE_CODES): Add h8sx_{unary,binary}_memory_operator.
* config/h8300/h8300.md: Add peepholes to combine reloads and
arithmetic insns.
2003-07-10 Richard Sandiford <rsandifo@redhat.com>
* config/h8300/h830.md (cmpqi): Use 'i' rather than 'n' in constraints.
(*cmphi_h8300hs, *addqi3, *addhi3_h8sx, subhi3): Likewise.
(and?i, ior?i, xor?i): Likewise.
2003-07-10 Richard Sandiford <rsandifo@redhat.com>
* config/h8300/h8300.c: Move enums and prototypes to head of file.
Various whitespace fixes.
(h8300_constant_length): New function, split out from...
(h8300_displacement_size): ...here. Rename h8300_displacement_length.
(h8300_classify_operand): Use IN_RANGE.
(h8300_classify_operand): Use h8300_constant_length.
(h8300_short_move_mem_p): Tighten size check.
(h8sx_mergeable_memrefs_p): Tighten equality check.
2003-06-30 Richard Sandiford <rsandifo@redhat.com>
* config/h8300/h8300.h (TARGET_CPU_CPP_BUILTINS): Define __H8300SX__
for -msx.
* config/h8300/crti.asm: Use .h8300sx or .h8300sxn for -msx code.
* config/h8300/crtn.asm: Likewise.
* config/h8300/lib1funcs.asm: Likewise. Use 32-bit pointers
if __H8300SX__ is defined.
2003-06-27 Richard Sandiford <rsandifo@redhat.com>
* config/h8300/h8300-protos.h (h8300_get_index): Add mode parameter.
* config/h8300/h8300.h (GO_IF_LEGITIMATE_ADDRESS): Update accordingly.
(GO_IF_MODE_DEPENDENT_ADDRESS): Treat POST_DEC, PRE_INC and indexed
addresses as mode-dependent.
* config/h8300/h8300.c (print_operand_address): Update call to
h8300_get_index.
(h8300_get_index): Take a mode argument. Rework to fix an
earlier misunderstanding.
2003-06-26 Richard Sandiford <rsandifo@redhat.com>
* config/h8300/h8300.c (zero_extendqisi2): Force the source operand
into a register if TARGET_H8300SX.
(*zero_extendqisi2_h8300hs, *extendqisi2_h8300): Disable for
TARGET_H8300SX. Also disable related define_splits.
(*zero_extendqisi2_h8sx, *extendqisi2_h8sx): New patterns.
2003-06-23 Richard Sandiford <rsandifo@redhat.com>
* config/h8300/h8300.c (h8300_rtx_costs): Add h8sx handling.
2003-06-20 Richard Sandiford <rsandifo@redhat.com>
* config/h8300/h8300.h (OK_FOR_Z): New macro.
(EXTRA_CONSTRAINT_STR): Check it.
* config/h8300/h8300.c (h8300_classify_operand): Accept null
class arguments.
(h8300_insn_length_from_table): Handle LENGTH_TABLE_MOV_IMM4.
* config/h8300/h8300.md (length_table): Add mov_imm4.
(movqi, movhi): Add Z <- W4 alternatives to h8sx patterns.
2003-06-20 Richard Sandiford <rsandifo@redhat.com>
* genattrtab.c (write_eligible_delay): Allow candidate_insn to
be a label.
* config/h8300/h8300.h (DELAY_SLOT_LENGTH): New macro.
* config/h8300/h8300.c (h8300_reorg): New function.
(TARGET_MACHINE_DEPENDENT_REORG): Define.
* config/h8300/h8300.md (length): Subtract the length of the
delay slot from (pc) when checking the range of forward branches.
(delay_slot, can_delay): New attributes.
(define_delay): Add bra/s handling.
(movmd_internal, return_h8sx, *return_1): Set can_delay to no.
(jump): Add delayed-branch handling.
2003-06-17 Richard Sandiford <rsandifo@redhat.com>
* expr.c (expand_strcpy): New function.
* builtins.c (expand_builtin_strcpy): Fall back on expand_strcpy.
(expand_builtin_stpcpy): Likewise.
* config/h8300/h8300-protos.h (h8sx_split_movmd): Remove.
(h8300_swap_into_er6, h8300_swap_out_of_er6): Declare.
* config/h8300/h8300.c (h8300_reg_class_from_letter): Tweak 'd'
handling to improve register allocation for -fno-omit-frame-pointer.
(h8sx_split_movmd): Delete, moving er6 handling into...
(h8300_swap_into_er6, h8300_swap_out_of_er6): ...these new functions.
* config/h8300/h8300.md (UNSPEC_STPCPY): New unspec constant.
(movmd): Add calls to copy_rtx.
(movmd_internal): In the second alternative, allow the initial and
final destination registers to be different . Update the splitter
accordingly. Call h8300_swap_into_er6 and h8300_swap_out_of_er6
instead of h8sx_split_movmd.
(stpcpy, movsd): New expanders.
(movsd_internal): New define_insn.
2003-06-13 Richard Sandiford <rsandifo@redhat.com>
* config/h8300/h8300-protos.h (h8300_reg_class_from_letter): Declare.
(h8sx_emit_movmd, h8sx_split_movmd): Declare.
* config/h8300/h8300.h (reg_class): Add COUNTER_REGS, SOURCE_REGS
and DESTINATION_REGS.
(REG_CLASS_NAMES, REG_CLASS_CONTENTS): Update accordingly.
(REGNO_REG_CLASS): Map er4, er5 and er6 to the new classes.
(REG_CLASS_FROM_LETTER): Use h8300_reg_class_from_letter.
(h8300_move_ratio): Declare.
(MOVE_RATIO): Use it.
* config/h8300/h8300.c (h8300_move_ratio): New variable.
(h8300_init_once): Initialize it.
(h8300_reg_class_from_letter): New function.
(print_operand): Add an 'm' prefix for printing ".b", ".w" or ".l".
(h8sx_emit_movmd, h8sx_split_movmd): New functions.
* config/h8300/h8300.md (UNSPEC_MOVMD): New unspec constant.
(COUNTER_REG, SOURCE_REG, DESTINATION_REG): New register constants.
(movstrsi, movmd): New expanders.
(movmd_internal): New insn.
2003-06-06 Richard Sandiford <rsandifo@redhat.com>
* config/h8300/h8300.h (EXTRA_MEMORY_CONSTRAINT): Define.
2003-06-04 Richard Sandiford <rsandifo@redhat.com>
* config/h8300/elf.h (LINK_SPEC): Use -m h8300sxnelf for -msx -mn.
* config/h8300/h8300.c (asm_file_start): Use .h8300sxn likewise.
2003-06-03 Richard Sandiford <rsandifo@redhat.com>
* config/h8300/h8300.c (nibble_operand): Fix warning.
* config/h8300/h8300.md (movstricthi): Set adjust_length to no.
(movsi_h8sx): Likewise here and the normal h8sx movhi pattern.
(movsf_h8300h): Disable for TARGET_H8300SX.
2003-06-03 Richard Sandiford <rsandifo@redhat.com>
* config/h8300/h8300.h (PREDICATE_CODES): Add h8300_ldm_parallel,
h8300_stm_parallel and h8300_return_parallel.
* config/h8300/h8300.c (h8300_push_pop, h8300_stack_offset_p,
h8300_ldm_stm_regno, h8300_ldm_stm_parallel, h8300_ldm_parallel,
h8300_stm_parallel, h8300_return_parallel): New functions.
(h8300_expand_prologue): Don't enforce ldm/stm register alignment
if TARGET_H8300SX. Use h8300_push_pop.
(h8300_expand_epilogue): Likewise. Try to merge the return insn
and final pop when generating h8sx code. Always emit some form
of return insn.
* config/h8300/h8300.md: Don't enforce register alignment in
stm peepholes if TARGET_H8300SX.
(ldm_h8300s, stm_h8300s, return_h8sx): New patterns.
(ldm_h8300s_[234], stm_h8300_[234]): Disable.
(epilogue): Expect h8300_expand_epilogue to emit a return insn.
2003-06-03 Richard Sandiford <rsandifo@redhat.com>
* config/h8300/t-h8300 (MULTILIB_OPTIONS): Add a -msx multilib.
(MULTILIB_DIRNAMES): Add a directory for it.
(MULTILIB_MATCHES): Delete.
2003-05-28 Richard Sandiford <rsandifo@redhat.com>
* final.c (walk_alter_subreg): Handle addresses with subregs
inside a ZERO_EXTEND or AND.
* config/h8300/h8300-protos.h (h8300_get_index): Declare.
* config/h8300/h8300.h (INDEX_REG_CLASS): Set to GENERAL_REGS
if TARGET_H8300SX.
(GO_IF_LEGITIMATE_ADDRESS): Use h8300_get_index.
* config/h8300/h8300.c (print_operand_address): Handle @(dd,RnL.b),
@(dd,Rn.w) and @(dd,ERn.L).
(h8300_displacement_size): Take the whole address as argument.
(h8300_classify_operand, h8300_short_move_mem_p): Adjust accordingly.
2003-05-28 Richard Sandiford <rsandifo@redhat.com>
* config/mips/mips-protos.h (h8300_operands_match_p): Declare.
(h8sx_mergeable_memrefs_p): Declare.
* config/h8300/h8300.h (HAVE_POST_DECREMENT): Define to TARGET_H8300SX.
(HAVE_PRE_INCREMENT): Likewise.
(GO_IF_LEGITIMATE_ADDRESS): Accept pre/post increment/decrement
addresses for TARGET_H8300SX,
* config/h8300/h8300.c (print_operand_address): Deal with PRE_INC
and POST_DEC.
(movb_length_table, movl_length_table): New tables.
(movw_length_table): Define to movb_length_table.
(h8300_displacement_size): New, split out from...
(h8300_classify_address): ...here. Handle pre/post inc/dec.
(h8300_short_immediate_length): Allow H8OP_MEM_COMPLEX operands.
(h8300_insn_length_from_table): Add cases for movb, movw and movl.
(h8sx_mergeable_memrefs_p, h8300_operands_match_p): New functions.
(output_plussi): Use add.l #xx:3,Rn and sub.l #xx:3,Rn for h8sx.
(compute_plussi_length, compute_plussi_cc): Update accordingly.
(h8sx_unary_shift_operator): Get the mode from the operator.
(binary_shift_operator): Likewise.
* config/h8300/h8300.md: If a peephole2 applies gen_lowpart to
a memory reference, check whether the reference is offsettable.
(length_table): Add movb, movw and movl.
(movqi): Add new h8sx pattern. Don't force one operand to be a
register when generating h8sx code.
(movhi, movsi, movsf): Likewise.
(movstrictqi): Use the length_table attribute.
(movstricthi): Likewise. Add h8sx alternative for mov.w #xx:3,Rn.
(addqi3): Split into a define_expand and define_insn. Don't accept
memory operands in the expander. Use h8300_operands_match_p to
check for matching operands in the define_insn.
(subqi3, negqi2, one_cmplqi2): Likewise.
(add[hs]i3): Don't accept memory operands in the expander. Likewise
in any patterns that are unused in h8sx code. In the h8sx patterns,
use h8300_operands_match_p to check whether operands match.
(sub[hs]i3, and[hi]3, ior[hs]i3, xor[hs]i3, neg[hsi]3,
one_cmpl[hs]i3): Likewise.
(andqi3, iorqi3, xorqi3): Likewise. Don't call fix_bit_operand
in the expander.
2003-05-23 Richard Sandiford <rsandifo@redhat.com>
* config/h8300/h8300-protos.h (expand_a_shift): Return a bool.
(h8300_insn_length_from_table): Add a second parameter.
(output_h8sx_shift): Declare.
* config/h8300/h8300.h (OK_FOR_W, OK_FOR_Y): New macros.
(EXTRA_CONSTRAINT): Replace with...
(EXTRA_CONSTRAINT_STR): ...this. Use OK_FOR_W and OK_FOR_Y.
(CONSTRAINT_LEN): Define, returning 2 for 'W' and 'Y'.
(PREDICATE_CODES): Add entries for h8sx_unary_shift_operator
and h8sx_binary_shift_operator.
* config/h8300/h8300.c (two_insn_adds_subs_operand): Return false
for TARGET_H8300SX.
(bit_operand): Replace use of EXTRA_CONSTRAINT with OK_FOR_U.
(bit_memory_operand, fix_bit_operand): Likewise.
(h8300_length_table_for_insn): Remove.
(h8300_classify_operand): Fix check for 16-bit operands in 32-bit
instructions.
(h8300_short_immediate_length, h8300_binary_length): New functions.
(h8300_insn_length_from_table): Add an opcodes parameter. Rework.
(output_plussi): Use sub to add negative constants.
(compute_plussi_length): Adjust accordingly.
(h8sx_single_shift_type): New enum.
(h8sx_single_shift, h8sx_unary_shift_operator,
h8sx_binary_shift_operator, output_h8sx_shift): New functions.
(expand_a_shift, expand_a_rotate): Emit nothing if the shift is a
single h8sx instruction. Return false in this case.
* config/h8300/h8300.md (length_table): Add short_immediate.
(length): Pass the operand array to h8300_insn_length_from_table.
(adjust_length): Assume "no" for insns with a length_table attribute.
(*cmphi_h8300hs, cmpsi): Add alternatives for #xx:3.
(*addhi3_h8300hs): Don't use for h8sx.
(*addhi3_h8sx): New pattern, with alternatives for add.w #xx:3
and sub.w #xx:3.
(ashl[qhs]i3, lshr[qhs]i3, ashr[qhs]i3, rotl[qhs]i3): Change operand
1's predicate to nonimmediate_operand. Only skip default expansion
if expand_a_shift or expand_a_rotate returns true. Add new patterns
for single h8sx shift instructions.
2003-05-22 Alexandre Oliva <aoliva@redhat.com>
* config/h8300/h8300.c (nibble_operand): Split out of...
(reg_or_nibble_operand): ... this.
* config/h8300/h8300.h (PREDICATE_CODES): Added nibble_operand.
* config/h8300/h8300.md: (mulqihi3, mulhisi3, umulqihi3,
umulhisi3): Introduce expand, and introduce separate insns for
sign- or zero-extended REG and already-extended CONST_INT.
2003-05-20 Richard Sandiford <rsandifo@redhat.com>
* config/h8300/h8300.c (h8300_unary_length): Fix miscounting.
* config/h8300/h8300.md (subqi3): Generalize for h8sx.
(subhi3): Likewise. Don't accept immediates for operand 1.
Remove the early clobber from second alternative of the h8300s pattern.
(subsi3): Generalize for h8sx. Force operand 2 into a register
on plain h8300 targets.
(subsi3_h8300): Use h8300_dst_operand for consistency with expander.
(subsi3_h8300h): Generalize for h8sx.
(one_cmplqi2, one_cmplhi2, one_cmplsi2): Likewise.
2003-05-19 Alexandre Oliva <aoliva@redhat.com>
* config/h8300/h8300.c (reg_or_nibble_operand): New.
* config/h8300/h8300.h (PREDICATE_CODES): Adjust.
(TARGET_H8300SXMUL): New.
(CONST_OK_FOR_P): New.
(CONST_OK_FOR_LETTER_P): Adjust.
* config/h8300/h8300.md (mulqihi3, mulhisi3, umulqihi3,
umulhisi3): Accept 4-bit immediate on H8SX.
(mulhi3, mulsi3, smulsi3_highpart, umulsi3_highpart): New.
(udivsi3, divhi3, udivsi3, divsi3): New.
2003-05-19 Richard Sandiford <rsandifo@redhat.com>
* config/h8300/h8300-protos.h (h8300_insn_length_from_table): Declare.
* config/h8300/h8300.h (OK_FOR_Q): New macro.
(EXTRA_CONSTRAINT): Use it to check the 'Q' constraint.
(PREDICATE_CODES): Add h8300_src_operand and h8300_dst_operand.
Add ADDRESSOF to the bit_operand entry.
* config/h8300/h8300.c (h8300_dst_operand): New predicate.
(h8300_src_operand): Likewise.
(bit_operand): Check nonimmediate_operand rather than general_operand.
Accept any nonimmediate_operand in h8sx code.
(h8300_and_costs): Initialize operands[1].
(h8300_rtx_costs) <AND>: Return false if the operands aren't valid.
(h8300_operand_class): New enum.
(h8300_length_table): New typedef.
(addb_length_table, addw_length_table, addl_length_table,
logicl_length_table): New tables.
(logicb_length_table, logicw_length_table): New macros.
(h8300_classify_operand, h8300_length_from_table,
h8300_length_table_for_insn, h8300_unary_length,
h8300_insn_length_from_table): New functions.
(output_plussi): Only use adds and subs for register destinations.
Disable redundant clause.
(compute_plussi_cc): Likewise.
(compute_plussi_length): Likewise. Use h8300_length_from_table
to work out the length of an insn.
(output_logical_op): Only use narrower immediate instructions
if the destination is a register.
(compute_logical_op_cc): Likewise.
(compute_logical_op_length): Likewise. Use h8300_length_from_table.
(h8300_adjust_insn_length): Tighten check for reg<->mem moves.
* config/h8300/h8300.md (length_table): New attribute.
(length): When an instruction has a length_table attribute, use
h8300_insn_length_from_table to calculate its default length.
(cmpqi): Use h8300_dst_operand for the first operand and
h8300_src_operand for the second.
(cmphi, *cmphi_h8300hs, cmpsi, negqi2, neghi2, neghi2_h8300h, negsi2,
negsi2_h8300h, addqi3, addhi3, *addhi3_h8300, *addhi3_h8300hs, addsi3,
addsi_h8300, addsi_h8300h, andhi3, andsi3, iorhi3,
iorsi3, xorhi3, xorsi3): Likewise.
(andqi3): Use h8300_src_operand for operand 2. Adjust the condition
so that it allows any combination of operands for TARGET_H8300SX.
(iorqi3, xorqi3): Likewise.
(cmpqi): Use the length_table attribute.
(*cmphi_h8300hs, cmpsi, addqi, *addhi3_h8300hs, andqi3, iorqi3,
xorqi3, negqi2, neghi2_h8300h, negsi2_h8300h): Likewise.
(cmpqi): Add 'Q' constraint.
(*cmphi_h8300hs, cmpsi, addqi, *addhi3_h8300hs, addsi_h8300h, andqi3,
iorqi3, xorqi3, negqi2, neghi2_h8300h, negsi2_h8300h): Likewise.
2003-05-14 Richard Sandiford <rsandifo@redhat.com>
* config/h8300/h8300.h (MASK_H8300SX): New macro.
(TARGET_H8300S): True for both -ms and -msx.
(TARGET_H8300SX): New macro.
(TARGET_SWITCHES): Add entries for -msx and -mno-sx.
* config/h8300/h8300.c (asm_file_start): Write .h8300sx for -msx.
* config/h8300/elf.h (LINK_SPEC): Use -m h8300sxelf for -msx.
* config/h8300/t-h8300 (MULTILIB_MATCHES): Use -ms multilibs for -msx.
[Temporary change.]
2003-02-28 Alexandre Oliva <aoliva@redhat.com>
* config/h8300/h8300.h (SIZE_TYPE, PTRDIFF_TYPE): Use short with
16-bit pointers and 32-bit ints.
* config/h8300/h8300.h (LEGITIMATE_CONSTANT_P): Accept
CONST_DOUBLE with mode no wider than SImode.
* config/h8300/h8300.md (extendqisi2_h8300): Add constraints for
output operand.
2003-02-27 Alexandre Oliva <aoliva@redhat.com>
* config/h8300/h8300.c (general_operand_src): Match CONSTANT_P_RTX
or SUBREG thereof.
* config/h8300/h8300.h (PREDICATE_CODES): Adjust.
2003-02-22 Alexandre Oliva <aoliva@redhat.com>
* config/h8300/h8300.c (dosize): Truncate sign * size to Pmode.
From-SVN: r84257
2004-07-08 05:40:34 +02:00
|
|
|
#if defined (__H8300H__) || defined (__H8300S__) || defined (__H8300SX__)
|
1994-07-12 00:45:37 +02:00
|
|
|
#define PUSHP push.l
|
|
|
|
#define POPP pop.l
|
|
|
|
|
|
|
|
#define A0P er0
|
|
|
|
#define A1P er1
|
|
|
|
#define A2P er2
|
|
|
|
#define A3P er3
|
|
|
|
#define S0P er4
|
|
|
|
#define S1P er5
|
|
|
|
#define S2P er6
|
|
|
|
|
|
|
|
#define A0E e0
|
|
|
|
#define A1E e1
|
|
|
|
#define A2E e2
|
|
|
|
#define A3E e3
|
|
|
|
#endif
|
|
|
|
|
2000-06-11 22:21:02 +02:00
|
|
|
#ifdef __H8300H__
|
2003-05-30 21:26:14 +02:00
|
|
|
#ifdef __NORMAL_MODE__
|
|
|
|
.h8300hn
|
|
|
|
#else
|
2000-06-11 22:21:02 +02:00
|
|
|
.h8300h
|
|
|
|
#endif
|
2003-05-30 21:26:14 +02:00
|
|
|
#endif
|
2000-06-11 22:21:02 +02:00
|
|
|
|
|
|
|
#ifdef __H8300S__
|
2003-05-30 21:26:14 +02:00
|
|
|
#ifdef __NORMAL_MODE__
|
|
|
|
.h8300sn
|
|
|
|
#else
|
2000-06-11 22:21:02 +02:00
|
|
|
.h8300s
|
|
|
|
#endif
|
2003-05-30 21:26:14 +02:00
|
|
|
#endif
|
Introduce H8SX support.
* expr.c (expand_strcpy): Renamed and moved to...
* builtins.c (expand_movstr): ... here. Tweak.
(expand_builtin_strcpy): Adjust. Use movstr if len can't be
computed or has side effects.
(expand_builtin_stpcpy): Likewise. Use strcpy if return value is
unused, or if mempcpy fails. Adjust the return value in the
latter case. Use movstr if everything else fails.
* doc/md.texi (movstr): Document.
(movmemM, clrmemM): Fix explanation of memory block operands.
* config/h8300/h8300.md (stpcpy): Renamed to...
(movstr): ... this. Adjust.
2004-07-07 Alexandre Oliva <aoliva@redhat.com>
* config/h8300/h8300.md: Rename movstr*, except for movstrict*, to
movmem* and clrstr* to clrmem*.
2004-06-27 Alexandre Oliva <aoliva@redhat.com>
* config/h8300/h8300.c (h8300_reg_class_from_letter): Map 'D' to
GENERAL_REGS, always.
(h8300_swap_into_er6, h8300_swap_into_er6): Handle the case of
getting the stack pointer as addr.
* config/h8300/h8300.h (PREDICATE_CODES): Remove constant rtxes
from general_operand_dst.
* config/h8300/h8300.md (movmd_internal_normal): New, normal-mode
variant of...
(movmd_internal): ... this. Add modes to operands. Disparage `D'
instead of requiring it to match only before reload.
(stpcpy_internal_normal): New, normal-mode variant of...
(stpcpy_internal): ... this. Add modes to operands. Disparage
`D' instead of requiring it to match only before reload.
* config/h8300/h8300-protos.h (h8300_legitimate_address_p): Add
mode argument.
* config/h8300/h8300.h (GO_IF_LEGITIMATE_ADDRESS): Pass it to...
* config/h8300/h8300.c (h8300_legitimate_address_p): Pass it to
h8300_get_index.
* config/h8300/h8300.md (attr type): Add call.
(attr can_delay): If type is call, set it no.
(call, call_value): Set type to call.
2004-06-21 Alexandre Oliva <aoliva@redhat.com>
* config/h8300/h8300.md (logicalhi3_sn, logicalsi3_sn): New.
2004-06-16 Alexandre Oliva <aoliva@redhat.com>
* tree.c (get_narrower): Don't narrow integral types into
non-integral types.
* config/h8300/h8300.c (h8300_expand_epilogue): Initialize
frame_size *before* the first use.
* config/h8300/h8300.md (movstrictqi): Reintroduce post-increment
on input.
(peephole2): Don't widen instructions that push SP. Move
decrement of SP to the end of all stm-generating peepholes.
2003-07-24 Richard Sandiford <rsandifo@redhat.com>
* config/h8300/h8300.md (insv): Prefer to use AND to clear a bitfield
and OR to set it to all ones.
2003-07-24 Richard Sandiford <rsandifo@redhat.com>
* config/h8300/h8300.md (can_delay): Default to "no" for bit branches.
(call, call_value): Set can_delay to "no".
2003-07-22 Richard Sandiford <rsandifo@redhat.com>
* config/h8300/h8300.md (extzv): Make subreg check more robust.
2003-07-21 Richard Sandiford <rsandifo@redhat.com>
* config/h8300/h8300.md (*brabit): Remove.
* config/h8300/h8300.md (*brabc, *brabs): Remove mode from
zero_extract. Use bit_memory_operand as the predicate for
operand 1 and 'WU' as the constraint. Check the difference
between the base length and the final one when deciding which
type of branch to use.
2003-07-21 Richard Sandiford <rsandifo@redhat.com>
* config/h8300/h8300.md (extzv): Remove mode from operands 0 and 1.
Use convert_move to extend the result for TARGET_H8300SX. Check
for QImode memory references. Optimize the case where the
destination is a paradoxical subreg.
2003-07-21 Richard Sandiford <rsandifo@redhat.com>
* config/h8300/h8300.md (*movsf_h8sx): Add an r <- G alternative.
* config/h8300/h8300.md (andqi): Remove bclr from h8sx version.
2003-07-21 Richard Sandiford <rsandifo@redhat.com>
* config/h8300/h8300.md: Include mova.md
(length_table): Add mova and mova_zero.
* config/h8300/h8300.c (print_operand): Handle '%o'. Print a length
after all constant addresses for '%R', '%X', '%T' and '%S'.
(h8300_mova_length): New function.
(h8300_insn_length_from_table): Use it to handle mova and mova_zero.
* config/h8300/t-h8300 (mova.md): Generate from genmova.sh. Add to
dependencies for s-config, etc.
* config/h8300/gemova.sh: New file.
* config/h8300/mova.md: Generated.
2003-07-20 Alexandre Oliva <aoliva@redhat.com>
* config/h8300/h8300.c (h8300_bitfield_length): New.
(nibble_operand): Adjust.
(h8300_binary_length): Handle conditional binary op.
(h8300_insn_length_from_table): Handle bitfield and bitbranch.
* config/h8300/h8300.h: Change constraints W# and Y# to P#>X and
P#<X, respectively. The original P is now IP4>X. Introduced P#>0
and P#<0, unused so far. W and Y are now prefixes to multi-letter
constraints. WU is introduced as a variant of U that requires a
mem, and is therefore considered an EXTRA_MEMORY_CONSTRAINT.
* config/h8300/h8300.md (attr type): Added bitbranch.
(attr length_table): Added bitfield and bitbranch.
(attr length): Compute bitbranch length.
(andqi): Separate pattern for H8300SX. Use bfld for loading the
least-significant bit of a byte.
(brabit, brabc, brabs): New.
(insv, extzv): Emit bfst and bfld on H8300SX.
(bfld, bfst, seq, sne): New.
(bstzhireg, cmpstz, bstz, bistz): New.
(cmpcondbset, condbset, cmpcondbclr, condbclr): New.
(cmpcondbsetreg, condbsetreg, cmpcondbclrreg, condbclrreg): New.
2003-07-11 Richard Sandiford <rsandifo@redhat.com>
* config/h8300/h8300.c (h8sx_binary_memory_operator): New function.
(h8sx_unary_memory_operator): New function.
* config/h8300/h8300.h (EXTRA_MEMORY_CONSTRAINT): Disable.
(PREDICATE_CODES): Add h8sx_{unary,binary}_memory_operator.
* config/h8300/h8300.md: Add peepholes to combine reloads and
arithmetic insns.
2003-07-10 Richard Sandiford <rsandifo@redhat.com>
* config/h8300/h830.md (cmpqi): Use 'i' rather than 'n' in constraints.
(*cmphi_h8300hs, *addqi3, *addhi3_h8sx, subhi3): Likewise.
(and?i, ior?i, xor?i): Likewise.
2003-07-10 Richard Sandiford <rsandifo@redhat.com>
* config/h8300/h8300.c: Move enums and prototypes to head of file.
Various whitespace fixes.
(h8300_constant_length): New function, split out from...
(h8300_displacement_size): ...here. Rename h8300_displacement_length.
(h8300_classify_operand): Use IN_RANGE.
(h8300_classify_operand): Use h8300_constant_length.
(h8300_short_move_mem_p): Tighten size check.
(h8sx_mergeable_memrefs_p): Tighten equality check.
2003-06-30 Richard Sandiford <rsandifo@redhat.com>
* config/h8300/h8300.h (TARGET_CPU_CPP_BUILTINS): Define __H8300SX__
for -msx.
* config/h8300/crti.asm: Use .h8300sx or .h8300sxn for -msx code.
* config/h8300/crtn.asm: Likewise.
* config/h8300/lib1funcs.asm: Likewise. Use 32-bit pointers
if __H8300SX__ is defined.
2003-06-27 Richard Sandiford <rsandifo@redhat.com>
* config/h8300/h8300-protos.h (h8300_get_index): Add mode parameter.
* config/h8300/h8300.h (GO_IF_LEGITIMATE_ADDRESS): Update accordingly.
(GO_IF_MODE_DEPENDENT_ADDRESS): Treat POST_DEC, PRE_INC and indexed
addresses as mode-dependent.
* config/h8300/h8300.c (print_operand_address): Update call to
h8300_get_index.
(h8300_get_index): Take a mode argument. Rework to fix an
earlier misunderstanding.
2003-06-26 Richard Sandiford <rsandifo@redhat.com>
* config/h8300/h8300.c (zero_extendqisi2): Force the source operand
into a register if TARGET_H8300SX.
(*zero_extendqisi2_h8300hs, *extendqisi2_h8300): Disable for
TARGET_H8300SX. Also disable related define_splits.
(*zero_extendqisi2_h8sx, *extendqisi2_h8sx): New patterns.
2003-06-23 Richard Sandiford <rsandifo@redhat.com>
* config/h8300/h8300.c (h8300_rtx_costs): Add h8sx handling.
2003-06-20 Richard Sandiford <rsandifo@redhat.com>
* config/h8300/h8300.h (OK_FOR_Z): New macro.
(EXTRA_CONSTRAINT_STR): Check it.
* config/h8300/h8300.c (h8300_classify_operand): Accept null
class arguments.
(h8300_insn_length_from_table): Handle LENGTH_TABLE_MOV_IMM4.
* config/h8300/h8300.md (length_table): Add mov_imm4.
(movqi, movhi): Add Z <- W4 alternatives to h8sx patterns.
2003-06-20 Richard Sandiford <rsandifo@redhat.com>
* genattrtab.c (write_eligible_delay): Allow candidate_insn to
be a label.
* config/h8300/h8300.h (DELAY_SLOT_LENGTH): New macro.
* config/h8300/h8300.c (h8300_reorg): New function.
(TARGET_MACHINE_DEPENDENT_REORG): Define.
* config/h8300/h8300.md (length): Subtract the length of the
delay slot from (pc) when checking the range of forward branches.
(delay_slot, can_delay): New attributes.
(define_delay): Add bra/s handling.
(movmd_internal, return_h8sx, *return_1): Set can_delay to no.
(jump): Add delayed-branch handling.
2003-06-17 Richard Sandiford <rsandifo@redhat.com>
* expr.c (expand_strcpy): New function.
* builtins.c (expand_builtin_strcpy): Fall back on expand_strcpy.
(expand_builtin_stpcpy): Likewise.
* config/h8300/h8300-protos.h (h8sx_split_movmd): Remove.
(h8300_swap_into_er6, h8300_swap_out_of_er6): Declare.
* config/h8300/h8300.c (h8300_reg_class_from_letter): Tweak 'd'
handling to improve register allocation for -fno-omit-frame-pointer.
(h8sx_split_movmd): Delete, moving er6 handling into...
(h8300_swap_into_er6, h8300_swap_out_of_er6): ...these new functions.
* config/h8300/h8300.md (UNSPEC_STPCPY): New unspec constant.
(movmd): Add calls to copy_rtx.
(movmd_internal): In the second alternative, allow the initial and
final destination registers to be different . Update the splitter
accordingly. Call h8300_swap_into_er6 and h8300_swap_out_of_er6
instead of h8sx_split_movmd.
(stpcpy, movsd): New expanders.
(movsd_internal): New define_insn.
2003-06-13 Richard Sandiford <rsandifo@redhat.com>
* config/h8300/h8300-protos.h (h8300_reg_class_from_letter): Declare.
(h8sx_emit_movmd, h8sx_split_movmd): Declare.
* config/h8300/h8300.h (reg_class): Add COUNTER_REGS, SOURCE_REGS
and DESTINATION_REGS.
(REG_CLASS_NAMES, REG_CLASS_CONTENTS): Update accordingly.
(REGNO_REG_CLASS): Map er4, er5 and er6 to the new classes.
(REG_CLASS_FROM_LETTER): Use h8300_reg_class_from_letter.
(h8300_move_ratio): Declare.
(MOVE_RATIO): Use it.
* config/h8300/h8300.c (h8300_move_ratio): New variable.
(h8300_init_once): Initialize it.
(h8300_reg_class_from_letter): New function.
(print_operand): Add an 'm' prefix for printing ".b", ".w" or ".l".
(h8sx_emit_movmd, h8sx_split_movmd): New functions.
* config/h8300/h8300.md (UNSPEC_MOVMD): New unspec constant.
(COUNTER_REG, SOURCE_REG, DESTINATION_REG): New register constants.
(movstrsi, movmd): New expanders.
(movmd_internal): New insn.
2003-06-06 Richard Sandiford <rsandifo@redhat.com>
* config/h8300/h8300.h (EXTRA_MEMORY_CONSTRAINT): Define.
2003-06-04 Richard Sandiford <rsandifo@redhat.com>
* config/h8300/elf.h (LINK_SPEC): Use -m h8300sxnelf for -msx -mn.
* config/h8300/h8300.c (asm_file_start): Use .h8300sxn likewise.
2003-06-03 Richard Sandiford <rsandifo@redhat.com>
* config/h8300/h8300.c (nibble_operand): Fix warning.
* config/h8300/h8300.md (movstricthi): Set adjust_length to no.
(movsi_h8sx): Likewise here and the normal h8sx movhi pattern.
(movsf_h8300h): Disable for TARGET_H8300SX.
2003-06-03 Richard Sandiford <rsandifo@redhat.com>
* config/h8300/h8300.h (PREDICATE_CODES): Add h8300_ldm_parallel,
h8300_stm_parallel and h8300_return_parallel.
* config/h8300/h8300.c (h8300_push_pop, h8300_stack_offset_p,
h8300_ldm_stm_regno, h8300_ldm_stm_parallel, h8300_ldm_parallel,
h8300_stm_parallel, h8300_return_parallel): New functions.
(h8300_expand_prologue): Don't enforce ldm/stm register alignment
if TARGET_H8300SX. Use h8300_push_pop.
(h8300_expand_epilogue): Likewise. Try to merge the return insn
and final pop when generating h8sx code. Always emit some form
of return insn.
* config/h8300/h8300.md: Don't enforce register alignment in
stm peepholes if TARGET_H8300SX.
(ldm_h8300s, stm_h8300s, return_h8sx): New patterns.
(ldm_h8300s_[234], stm_h8300_[234]): Disable.
(epilogue): Expect h8300_expand_epilogue to emit a return insn.
2003-06-03 Richard Sandiford <rsandifo@redhat.com>
* config/h8300/t-h8300 (MULTILIB_OPTIONS): Add a -msx multilib.
(MULTILIB_DIRNAMES): Add a directory for it.
(MULTILIB_MATCHES): Delete.
2003-05-28 Richard Sandiford <rsandifo@redhat.com>
* final.c (walk_alter_subreg): Handle addresses with subregs
inside a ZERO_EXTEND or AND.
* config/h8300/h8300-protos.h (h8300_get_index): Declare.
* config/h8300/h8300.h (INDEX_REG_CLASS): Set to GENERAL_REGS
if TARGET_H8300SX.
(GO_IF_LEGITIMATE_ADDRESS): Use h8300_get_index.
* config/h8300/h8300.c (print_operand_address): Handle @(dd,RnL.b),
@(dd,Rn.w) and @(dd,ERn.L).
(h8300_displacement_size): Take the whole address as argument.
(h8300_classify_operand, h8300_short_move_mem_p): Adjust accordingly.
2003-05-28 Richard Sandiford <rsandifo@redhat.com>
* config/mips/mips-protos.h (h8300_operands_match_p): Declare.
(h8sx_mergeable_memrefs_p): Declare.
* config/h8300/h8300.h (HAVE_POST_DECREMENT): Define to TARGET_H8300SX.
(HAVE_PRE_INCREMENT): Likewise.
(GO_IF_LEGITIMATE_ADDRESS): Accept pre/post increment/decrement
addresses for TARGET_H8300SX,
* config/h8300/h8300.c (print_operand_address): Deal with PRE_INC
and POST_DEC.
(movb_length_table, movl_length_table): New tables.
(movw_length_table): Define to movb_length_table.
(h8300_displacement_size): New, split out from...
(h8300_classify_address): ...here. Handle pre/post inc/dec.
(h8300_short_immediate_length): Allow H8OP_MEM_COMPLEX operands.
(h8300_insn_length_from_table): Add cases for movb, movw and movl.
(h8sx_mergeable_memrefs_p, h8300_operands_match_p): New functions.
(output_plussi): Use add.l #xx:3,Rn and sub.l #xx:3,Rn for h8sx.
(compute_plussi_length, compute_plussi_cc): Update accordingly.
(h8sx_unary_shift_operator): Get the mode from the operator.
(binary_shift_operator): Likewise.
* config/h8300/h8300.md: If a peephole2 applies gen_lowpart to
a memory reference, check whether the reference is offsettable.
(length_table): Add movb, movw and movl.
(movqi): Add new h8sx pattern. Don't force one operand to be a
register when generating h8sx code.
(movhi, movsi, movsf): Likewise.
(movstrictqi): Use the length_table attribute.
(movstricthi): Likewise. Add h8sx alternative for mov.w #xx:3,Rn.
(addqi3): Split into a define_expand and define_insn. Don't accept
memory operands in the expander. Use h8300_operands_match_p to
check for matching operands in the define_insn.
(subqi3, negqi2, one_cmplqi2): Likewise.
(add[hs]i3): Don't accept memory operands in the expander. Likewise
in any patterns that are unused in h8sx code. In the h8sx patterns,
use h8300_operands_match_p to check whether operands match.
(sub[hs]i3, and[hi]3, ior[hs]i3, xor[hs]i3, neg[hsi]3,
one_cmpl[hs]i3): Likewise.
(andqi3, iorqi3, xorqi3): Likewise. Don't call fix_bit_operand
in the expander.
2003-05-23 Richard Sandiford <rsandifo@redhat.com>
* config/h8300/h8300-protos.h (expand_a_shift): Return a bool.
(h8300_insn_length_from_table): Add a second parameter.
(output_h8sx_shift): Declare.
* config/h8300/h8300.h (OK_FOR_W, OK_FOR_Y): New macros.
(EXTRA_CONSTRAINT): Replace with...
(EXTRA_CONSTRAINT_STR): ...this. Use OK_FOR_W and OK_FOR_Y.
(CONSTRAINT_LEN): Define, returning 2 for 'W' and 'Y'.
(PREDICATE_CODES): Add entries for h8sx_unary_shift_operator
and h8sx_binary_shift_operator.
* config/h8300/h8300.c (two_insn_adds_subs_operand): Return false
for TARGET_H8300SX.
(bit_operand): Replace use of EXTRA_CONSTRAINT with OK_FOR_U.
(bit_memory_operand, fix_bit_operand): Likewise.
(h8300_length_table_for_insn): Remove.
(h8300_classify_operand): Fix check for 16-bit operands in 32-bit
instructions.
(h8300_short_immediate_length, h8300_binary_length): New functions.
(h8300_insn_length_from_table): Add an opcodes parameter. Rework.
(output_plussi): Use sub to add negative constants.
(compute_plussi_length): Adjust accordingly.
(h8sx_single_shift_type): New enum.
(h8sx_single_shift, h8sx_unary_shift_operator,
h8sx_binary_shift_operator, output_h8sx_shift): New functions.
(expand_a_shift, expand_a_rotate): Emit nothing if the shift is a
single h8sx instruction. Return false in this case.
* config/h8300/h8300.md (length_table): Add short_immediate.
(length): Pass the operand array to h8300_insn_length_from_table.
(adjust_length): Assume "no" for insns with a length_table attribute.
(*cmphi_h8300hs, cmpsi): Add alternatives for #xx:3.
(*addhi3_h8300hs): Don't use for h8sx.
(*addhi3_h8sx): New pattern, with alternatives for add.w #xx:3
and sub.w #xx:3.
(ashl[qhs]i3, lshr[qhs]i3, ashr[qhs]i3, rotl[qhs]i3): Change operand
1's predicate to nonimmediate_operand. Only skip default expansion
if expand_a_shift or expand_a_rotate returns true. Add new patterns
for single h8sx shift instructions.
2003-05-22 Alexandre Oliva <aoliva@redhat.com>
* config/h8300/h8300.c (nibble_operand): Split out of...
(reg_or_nibble_operand): ... this.
* config/h8300/h8300.h (PREDICATE_CODES): Added nibble_operand.
* config/h8300/h8300.md: (mulqihi3, mulhisi3, umulqihi3,
umulhisi3): Introduce expand, and introduce separate insns for
sign- or zero-extended REG and already-extended CONST_INT.
2003-05-20 Richard Sandiford <rsandifo@redhat.com>
* config/h8300/h8300.c (h8300_unary_length): Fix miscounting.
* config/h8300/h8300.md (subqi3): Generalize for h8sx.
(subhi3): Likewise. Don't accept immediates for operand 1.
Remove the early clobber from second alternative of the h8300s pattern.
(subsi3): Generalize for h8sx. Force operand 2 into a register
on plain h8300 targets.
(subsi3_h8300): Use h8300_dst_operand for consistency with expander.
(subsi3_h8300h): Generalize for h8sx.
(one_cmplqi2, one_cmplhi2, one_cmplsi2): Likewise.
2003-05-19 Alexandre Oliva <aoliva@redhat.com>
* config/h8300/h8300.c (reg_or_nibble_operand): New.
* config/h8300/h8300.h (PREDICATE_CODES): Adjust.
(TARGET_H8300SXMUL): New.
(CONST_OK_FOR_P): New.
(CONST_OK_FOR_LETTER_P): Adjust.
* config/h8300/h8300.md (mulqihi3, mulhisi3, umulqihi3,
umulhisi3): Accept 4-bit immediate on H8SX.
(mulhi3, mulsi3, smulsi3_highpart, umulsi3_highpart): New.
(udivsi3, divhi3, udivsi3, divsi3): New.
2003-05-19 Richard Sandiford <rsandifo@redhat.com>
* config/h8300/h8300-protos.h (h8300_insn_length_from_table): Declare.
* config/h8300/h8300.h (OK_FOR_Q): New macro.
(EXTRA_CONSTRAINT): Use it to check the 'Q' constraint.
(PREDICATE_CODES): Add h8300_src_operand and h8300_dst_operand.
Add ADDRESSOF to the bit_operand entry.
* config/h8300/h8300.c (h8300_dst_operand): New predicate.
(h8300_src_operand): Likewise.
(bit_operand): Check nonimmediate_operand rather than general_operand.
Accept any nonimmediate_operand in h8sx code.
(h8300_and_costs): Initialize operands[1].
(h8300_rtx_costs) <AND>: Return false if the operands aren't valid.
(h8300_operand_class): New enum.
(h8300_length_table): New typedef.
(addb_length_table, addw_length_table, addl_length_table,
logicl_length_table): New tables.
(logicb_length_table, logicw_length_table): New macros.
(h8300_classify_operand, h8300_length_from_table,
h8300_length_table_for_insn, h8300_unary_length,
h8300_insn_length_from_table): New functions.
(output_plussi): Only use adds and subs for register destinations.
Disable redundant clause.
(compute_plussi_cc): Likewise.
(compute_plussi_length): Likewise. Use h8300_length_from_table
to work out the length of an insn.
(output_logical_op): Only use narrower immediate instructions
if the destination is a register.
(compute_logical_op_cc): Likewise.
(compute_logical_op_length): Likewise. Use h8300_length_from_table.
(h8300_adjust_insn_length): Tighten check for reg<->mem moves.
* config/h8300/h8300.md (length_table): New attribute.
(length): When an instruction has a length_table attribute, use
h8300_insn_length_from_table to calculate its default length.
(cmpqi): Use h8300_dst_operand for the first operand and
h8300_src_operand for the second.
(cmphi, *cmphi_h8300hs, cmpsi, negqi2, neghi2, neghi2_h8300h, negsi2,
negsi2_h8300h, addqi3, addhi3, *addhi3_h8300, *addhi3_h8300hs, addsi3,
addsi_h8300, addsi_h8300h, andhi3, andsi3, iorhi3,
iorsi3, xorhi3, xorsi3): Likewise.
(andqi3): Use h8300_src_operand for operand 2. Adjust the condition
so that it allows any combination of operands for TARGET_H8300SX.
(iorqi3, xorqi3): Likewise.
(cmpqi): Use the length_table attribute.
(*cmphi_h8300hs, cmpsi, addqi, *addhi3_h8300hs, andqi3, iorqi3,
xorqi3, negqi2, neghi2_h8300h, negsi2_h8300h): Likewise.
(cmpqi): Add 'Q' constraint.
(*cmphi_h8300hs, cmpsi, addqi, *addhi3_h8300hs, addsi_h8300h, andqi3,
iorqi3, xorqi3, negqi2, neghi2_h8300h, negsi2_h8300h): Likewise.
2003-05-14 Richard Sandiford <rsandifo@redhat.com>
* config/h8300/h8300.h (MASK_H8300SX): New macro.
(TARGET_H8300S): True for both -ms and -msx.
(TARGET_H8300SX): New macro.
(TARGET_SWITCHES): Add entries for -msx and -mno-sx.
* config/h8300/h8300.c (asm_file_start): Write .h8300sx for -msx.
* config/h8300/elf.h (LINK_SPEC): Use -m h8300sxelf for -msx.
* config/h8300/t-h8300 (MULTILIB_MATCHES): Use -ms multilibs for -msx.
[Temporary change.]
2003-02-28 Alexandre Oliva <aoliva@redhat.com>
* config/h8300/h8300.h (SIZE_TYPE, PTRDIFF_TYPE): Use short with
16-bit pointers and 32-bit ints.
* config/h8300/h8300.h (LEGITIMATE_CONSTANT_P): Accept
CONST_DOUBLE with mode no wider than SImode.
* config/h8300/h8300.md (extendqisi2_h8300): Add constraints for
output operand.
2003-02-27 Alexandre Oliva <aoliva@redhat.com>
* config/h8300/h8300.c (general_operand_src): Match CONSTANT_P_RTX
or SUBREG thereof.
* config/h8300/h8300.h (PREDICATE_CODES): Adjust.
2003-02-22 Alexandre Oliva <aoliva@redhat.com>
* config/h8300/h8300.c (dosize): Truncate sign * size to Pmode.
From-SVN: r84257
2004-07-08 05:40:34 +02:00
|
|
|
#ifdef __H8300SX__
|
|
|
|
#ifdef __NORMAL_MODE__
|
|
|
|
.h8300sxn
|
|
|
|
#else
|
|
|
|
.h8300sx
|
|
|
|
#endif
|
|
|
|
#endif
|
2000-06-11 22:21:02 +02:00
|
|
|
|
1994-07-12 00:45:37 +02:00
|
|
|
#ifdef L_cmpsi2
|
|
|
|
#ifdef __H8300__
|
|
|
|
.section .text
|
|
|
|
.align 2
|
|
|
|
.global ___cmpsi2
|
|
|
|
___cmpsi2:
|
2002-04-20 18:02:36 +02:00
|
|
|
cmp.w A0,A2
|
1994-07-12 00:45:37 +02:00
|
|
|
bne .L2
|
2002-04-20 18:02:36 +02:00
|
|
|
cmp.w A1,A3
|
|
|
|
bne .L4
|
1994-07-12 00:45:37 +02:00
|
|
|
mov.w #1,A0
|
|
|
|
rts
|
|
|
|
.L2:
|
2002-04-20 18:02:36 +02:00
|
|
|
bgt .L5
|
1994-07-12 00:45:37 +02:00
|
|
|
.L3:
|
|
|
|
mov.w #2,A0
|
2002-04-20 18:02:36 +02:00
|
|
|
rts
|
|
|
|
.L4:
|
|
|
|
bls .L3
|
1994-07-12 00:45:37 +02:00
|
|
|
.L5:
|
2002-04-20 18:02:36 +02:00
|
|
|
sub.w A0,A0
|
1994-07-12 00:45:37 +02:00
|
|
|
rts
|
|
|
|
.end
|
|
|
|
#endif
|
|
|
|
#endif /* L_cmpsi2 */
|
|
|
|
|
|
|
|
#ifdef L_ucmpsi2
|
|
|
|
#ifdef __H8300__
|
|
|
|
.section .text
|
|
|
|
.align 2
|
|
|
|
.global ___ucmpsi2
|
|
|
|
___ucmpsi2:
|
2002-04-20 18:02:36 +02:00
|
|
|
cmp.w A0,A2
|
1994-07-12 00:45:37 +02:00
|
|
|
bne .L2
|
2002-04-20 18:02:36 +02:00
|
|
|
cmp.w A1,A3
|
|
|
|
bne .L4
|
1994-07-12 00:45:37 +02:00
|
|
|
mov.w #1,A0
|
|
|
|
rts
|
|
|
|
.L2:
|
2002-04-20 18:02:36 +02:00
|
|
|
bhi .L5
|
1994-07-12 00:45:37 +02:00
|
|
|
.L3:
|
|
|
|
mov.w #2,A0
|
2002-04-20 18:02:36 +02:00
|
|
|
rts
|
|
|
|
.L4:
|
|
|
|
bls .L3
|
1994-07-12 00:45:37 +02:00
|
|
|
.L5:
|
2002-04-20 18:02:36 +02:00
|
|
|
sub.w A0,A0
|
1994-07-12 00:45:37 +02:00
|
|
|
rts
|
|
|
|
.end
|
|
|
|
#endif
|
|
|
|
#endif /* L_ucmpsi2 */
|
|
|
|
|
|
|
|
#ifdef L_divhi3
|
|
|
|
|
|
|
|
;; HImode divides for the H8/300.
|
|
|
|
;; We bunch all of this into one object file since there are several
|
|
|
|
;; "supporting routines".
|
|
|
|
|
|
|
|
; general purpose normalize routine
|
2002-05-03 14:28:41 +02:00
|
|
|
;
|
1994-07-12 00:45:37 +02:00
|
|
|
; divisor in A0
|
|
|
|
; dividend in A1
|
|
|
|
; turns both into +ve numbers, and leaves what the answer sign
|
|
|
|
; should be in A2L
|
|
|
|
|
|
|
|
#ifdef __H8300__
|
|
|
|
.section .text
|
|
|
|
.align 2
|
|
|
|
divnorm:
|
|
|
|
or A0H,A0H ; is divisor > 0
|
2004-01-25 18:51:51 +01:00
|
|
|
stc ccr,A2L
|
2002-05-03 14:28:41 +02:00
|
|
|
bge _lab1
|
1994-07-12 00:45:37 +02:00
|
|
|
not A0H ; no - then make it +ve
|
|
|
|
not A0L
|
2002-05-03 14:28:41 +02:00
|
|
|
adds #1,A0
|
1994-07-12 00:45:37 +02:00
|
|
|
_lab1: or A1H,A1H ; look at dividend
|
2002-05-03 14:28:41 +02:00
|
|
|
bge _lab2
|
1994-07-12 00:45:37 +02:00
|
|
|
not A1H ; it is -ve, make it positive
|
|
|
|
not A1L
|
|
|
|
adds #1,A1
|
2004-01-25 18:51:51 +01:00
|
|
|
xor #0x8,A2L; and toggle sign of result
|
1994-07-12 00:45:37 +02:00
|
|
|
_lab2: rts
|
1996-04-11 22:10:57 +02:00
|
|
|
;; Basically the same, except that the sign of the divisor determines
|
|
|
|
;; the sign.
|
|
|
|
modnorm:
|
|
|
|
or A0H,A0H ; is divisor > 0
|
2004-01-25 18:51:51 +01:00
|
|
|
stc ccr,A2L
|
2002-05-03 14:28:41 +02:00
|
|
|
bge _lab7
|
1996-04-11 22:10:57 +02:00
|
|
|
not A0H ; no - then make it +ve
|
|
|
|
not A0L
|
2002-05-03 14:28:41 +02:00
|
|
|
adds #1,A0
|
1996-04-11 22:10:57 +02:00
|
|
|
_lab7: or A1H,A1H ; look at dividend
|
2002-05-03 14:28:41 +02:00
|
|
|
bge _lab8
|
1996-04-11 22:10:57 +02:00
|
|
|
not A1H ; it is -ve, make it positive
|
|
|
|
not A1L
|
|
|
|
adds #1,A1
|
|
|
|
_lab8: rts
|
1994-07-12 00:45:37 +02:00
|
|
|
|
|
|
|
; A0=A0/A1 signed
|
|
|
|
|
|
|
|
.global ___divhi3
|
|
|
|
___divhi3:
|
|
|
|
bsr divnorm
|
|
|
|
bsr ___udivhi3
|
2004-01-25 18:51:51 +01:00
|
|
|
negans: btst #3,A2L ; should answer be negative ?
|
1994-07-12 00:45:37 +02:00
|
|
|
beq _lab4
|
|
|
|
not A0H ; yes, so make it so
|
|
|
|
not A0L
|
|
|
|
adds #1,A0
|
2002-05-03 14:28:41 +02:00
|
|
|
_lab4: rts
|
1994-07-12 00:45:37 +02:00
|
|
|
|
|
|
|
; A0=A0%A1 signed
|
|
|
|
|
|
|
|
.global ___modhi3
|
|
|
|
___modhi3:
|
1996-04-11 22:10:57 +02:00
|
|
|
bsr modnorm
|
1994-07-12 00:45:37 +02:00
|
|
|
bsr ___udivhi3
|
|
|
|
mov A3,A0
|
|
|
|
bra negans
|
|
|
|
|
|
|
|
; A0=A0%A1 unsigned
|
|
|
|
|
|
|
|
.global ___umodhi3
|
|
|
|
___umodhi3:
|
|
|
|
bsr ___udivhi3
|
|
|
|
mov A3,A0
|
|
|
|
rts
|
|
|
|
|
|
|
|
; A0=A0/A1 unsigned
|
|
|
|
; A3=A0%A1 unsigned
|
|
|
|
; A2H trashed
|
|
|
|
; D high 8 bits of denom
|
|
|
|
; d low 8 bits of denom
|
|
|
|
; N high 8 bits of num
|
|
|
|
; n low 8 bits of num
|
|
|
|
; M high 8 bits of mod
|
|
|
|
; m low 8 bits of mod
|
|
|
|
; Q high 8 bits of quot
|
|
|
|
; q low 8 bits of quot
|
|
|
|
; P preserve
|
|
|
|
|
2001-08-29 16:45:35 +02:00
|
|
|
; The H8/300 only has a 16/8 bit divide, so we look at the incoming and
|
1994-07-12 00:45:37 +02:00
|
|
|
; see how to partition up the expression.
|
|
|
|
|
|
|
|
.global ___udivhi3
|
|
|
|
___udivhi3:
|
2002-05-03 14:28:41 +02:00
|
|
|
; A0 A1 A2 A3
|
1994-07-12 00:45:37 +02:00
|
|
|
; Nn Dd P
|
2002-05-03 14:28:41 +02:00
|
|
|
sub.w A3,A3 ; Nn Dd xP 00
|
|
|
|
or A1H,A1H
|
1994-07-12 00:45:37 +02:00
|
|
|
bne divlongway
|
2002-05-03 14:28:41 +02:00
|
|
|
or A0H,A0H
|
|
|
|
beq _lab6
|
1994-07-12 00:45:37 +02:00
|
|
|
|
|
|
|
; we know that D == 0 and N is != 0
|
|
|
|
mov.b A0H,A3L ; Nn Dd xP 0N
|
|
|
|
divxu A1L,A3 ; MQ
|
|
|
|
mov.b A3L,A0H ; Q
|
|
|
|
; dealt with N, do n
|
|
|
|
_lab6: mov.b A0L,A3L ; n
|
|
|
|
divxu A1L,A3 ; mq
|
|
|
|
mov.b A3L,A0L ; Qq
|
|
|
|
mov.b A3H,A3L ; m
|
|
|
|
mov.b #0x0,A3H ; Qq 0m
|
2002-05-03 14:28:41 +02:00
|
|
|
rts
|
1994-07-12 00:45:37 +02:00
|
|
|
|
|
|
|
; D != 0 - which means the denominator is
|
|
|
|
; loop around to get the result.
|
|
|
|
|
|
|
|
divlongway:
|
|
|
|
mov.b A0H,A3L ; Nn Dd xP 0N
|
|
|
|
mov.b #0x0,A0H ; high byte of answer has to be zero
|
|
|
|
mov.b #0x8,A2H ; 8
|
|
|
|
div8: add.b A0L,A0L ; n*=2
|
|
|
|
rotxl A3L ; Make remainder bigger
|
2002-05-03 14:28:41 +02:00
|
|
|
rotxl A3H
|
1994-07-12 00:45:37 +02:00
|
|
|
sub.w A1,A3 ; Q-=N
|
|
|
|
bhs setbit ; set a bit ?
|
|
|
|
add.w A1,A3 ; no : too far , Q+=N
|
|
|
|
|
2002-05-03 14:28:41 +02:00
|
|
|
dec A2H
|
|
|
|
bne div8 ; next bit
|
|
|
|
rts
|
1994-07-12 00:45:37 +02:00
|
|
|
|
|
|
|
setbit: inc A0L ; do insert bit
|
2002-05-03 14:28:41 +02:00
|
|
|
dec A2H
|
|
|
|
bne div8 ; next bit
|
|
|
|
rts
|
1994-07-12 00:45:37 +02:00
|
|
|
|
|
|
|
#endif /* __H8300__ */
|
|
|
|
#endif /* L_divhi3 */
|
|
|
|
|
|
|
|
#ifdef L_divsi3
|
|
|
|
|
|
|
|
;; 4 byte integer divides for the H8/300.
|
|
|
|
;;
|
2002-05-03 14:28:41 +02:00
|
|
|
;; We have one routine which does all the work and lots of
|
1994-07-12 00:45:37 +02:00
|
|
|
;; little ones which prepare the args and massage the sign.
|
|
|
|
;; We bunch all of this into one object file since there are several
|
|
|
|
;; "supporting routines".
|
|
|
|
|
|
|
|
.section .text
|
|
|
|
.align 2
|
|
|
|
|
|
|
|
; Put abs SIs into r0/r1 and r2/r3, and leave a 1 in r6l with sign of rest.
|
|
|
|
; This function is here to keep branch displacements small.
|
|
|
|
|
|
|
|
#ifdef __H8300__
|
|
|
|
|
|
|
|
divnorm:
|
|
|
|
mov.b A0H,A0H ; is the numerator -ve
|
2003-11-15 22:36:49 +01:00
|
|
|
stc ccr,S2L ; keep the sign in bit 3 of S2L
|
1994-07-12 00:45:37 +02:00
|
|
|
bge postive
|
|
|
|
|
|
|
|
; negate arg
|
|
|
|
not A0H
|
|
|
|
not A1H
|
|
|
|
not A0L
|
|
|
|
not A1L
|
|
|
|
|
|
|
|
add #1,A1L
|
|
|
|
addx #0,A1H
|
|
|
|
addx #0,A0L
|
1997-04-30 00:12:25 +02:00
|
|
|
addx #0,A0H
|
1994-07-12 00:45:37 +02:00
|
|
|
postive:
|
|
|
|
mov.b A2H,A2H ; is the denominator -ve
|
|
|
|
bge postive2
|
2002-05-03 14:28:41 +02:00
|
|
|
not A2L
|
1994-07-12 00:45:37 +02:00
|
|
|
not A2H
|
|
|
|
not A3L
|
|
|
|
not A3H
|
2002-05-03 14:28:41 +02:00
|
|
|
add.b #1,A3L
|
1994-07-12 00:45:37 +02:00
|
|
|
addx #0,A3H
|
|
|
|
addx #0,A2L
|
|
|
|
addx #0,A2H
|
2003-11-15 22:36:49 +01:00
|
|
|
xor.b #0x08,S2L ; toggle the result sign
|
1994-07-12 00:45:37 +02:00
|
|
|
postive2:
|
|
|
|
rts
|
|
|
|
|
1996-04-11 22:10:57 +02:00
|
|
|
;; Basically the same, except that the sign of the divisor determines
|
|
|
|
;; the sign.
|
|
|
|
modnorm:
|
|
|
|
mov.b A0H,A0H ; is the numerator -ve
|
2003-11-15 22:36:49 +01:00
|
|
|
stc ccr,S2L ; keep the sign in bit 3 of S2L
|
1996-04-11 22:10:57 +02:00
|
|
|
bge mpostive
|
|
|
|
|
|
|
|
; negate arg
|
|
|
|
not A0H
|
|
|
|
not A1H
|
|
|
|
not A0L
|
|
|
|
not A1L
|
|
|
|
|
|
|
|
add #1,A1L
|
|
|
|
addx #0,A1H
|
|
|
|
addx #0,A0L
|
1997-04-30 00:12:25 +02:00
|
|
|
addx #0,A0H
|
1996-04-11 22:10:57 +02:00
|
|
|
mpostive:
|
|
|
|
mov.b A2H,A2H ; is the denominator -ve
|
|
|
|
bge mpostive2
|
2002-05-03 14:28:41 +02:00
|
|
|
not A2L
|
1996-04-11 22:10:57 +02:00
|
|
|
not A2H
|
|
|
|
not A3L
|
|
|
|
not A3H
|
2002-05-03 14:28:41 +02:00
|
|
|
add.b #1,A3L
|
1996-04-11 22:10:57 +02:00
|
|
|
addx #0,A3H
|
|
|
|
addx #0,A2L
|
|
|
|
addx #0,A2H
|
|
|
|
mpostive2:
|
|
|
|
rts
|
|
|
|
|
1994-07-12 00:45:37 +02:00
|
|
|
#else /* __H8300H__ */
|
|
|
|
|
|
|
|
divnorm:
|
|
|
|
mov.l A0P,A0P ; is the numerator -ve
|
2003-11-15 22:36:49 +01:00
|
|
|
stc ccr,S2L ; keep the sign in bit 3 of S2L
|
1994-07-12 00:45:37 +02:00
|
|
|
bge postive
|
|
|
|
|
|
|
|
neg.l A0P ; negate arg
|
|
|
|
|
|
|
|
postive:
|
|
|
|
mov.l A1P,A1P ; is the denominator -ve
|
|
|
|
bge postive2
|
|
|
|
|
|
|
|
neg.l A1P ; negate arg
|
2003-11-15 22:36:49 +01:00
|
|
|
xor.b #0x08,S2L ; toggle the result sign
|
1994-07-12 00:45:37 +02:00
|
|
|
|
|
|
|
postive2:
|
|
|
|
rts
|
|
|
|
|
1996-04-11 22:10:57 +02:00
|
|
|
;; Basically the same, except that the sign of the divisor determines
|
|
|
|
;; the sign.
|
|
|
|
modnorm:
|
|
|
|
mov.l A0P,A0P ; is the numerator -ve
|
2003-11-15 22:36:49 +01:00
|
|
|
stc ccr,S2L ; keep the sign in bit 3 of S2L
|
1996-04-11 22:10:57 +02:00
|
|
|
bge mpostive
|
|
|
|
|
|
|
|
neg.l A0P ; negate arg
|
|
|
|
|
|
|
|
mpostive:
|
|
|
|
mov.l A1P,A1P ; is the denominator -ve
|
|
|
|
bge mpostive2
|
|
|
|
|
|
|
|
neg.l A1P ; negate arg
|
|
|
|
|
|
|
|
mpostive2:
|
|
|
|
rts
|
|
|
|
|
1994-07-12 00:45:37 +02:00
|
|
|
#endif
|
|
|
|
|
|
|
|
; numerator in A0/A1
|
|
|
|
; denominator in A2/A3
|
|
|
|
.global ___modsi3
|
|
|
|
___modsi3:
|
2003-11-14 13:39:27 +01:00
|
|
|
#ifdef __H8300__
|
2002-05-03 14:28:41 +02:00
|
|
|
PUSHP S2P
|
1994-07-12 00:45:37 +02:00
|
|
|
PUSHP S0P
|
|
|
|
PUSHP S1P
|
1996-04-11 22:10:57 +02:00
|
|
|
bsr modnorm
|
2003-11-14 05:35:46 +01:00
|
|
|
bsr divmodsi4
|
1994-07-12 00:45:37 +02:00
|
|
|
mov S0,A0
|
|
|
|
mov S1,A1
|
2003-11-14 13:39:27 +01:00
|
|
|
bra exitdiv
|
1994-07-12 00:45:37 +02:00
|
|
|
#else
|
2003-11-14 13:39:27 +01:00
|
|
|
PUSHP S2P
|
|
|
|
bsr modnorm
|
2003-11-14 05:35:46 +01:00
|
|
|
bsr ___udivsi3
|
2003-11-14 03:06:37 +01:00
|
|
|
mov.l er3,er0
|
1994-07-12 00:45:37 +02:00
|
|
|
bra exitdiv
|
2003-11-14 13:39:27 +01:00
|
|
|
#endif
|
1994-07-12 00:45:37 +02:00
|
|
|
|
2003-11-14 05:35:46 +01:00
|
|
|
;; H8/300H and H8S version of ___udivsi3 is defined later in
|
|
|
|
;; the file.
|
|
|
|
#ifdef __H8300__
|
1994-07-12 00:45:37 +02:00
|
|
|
.global ___udivsi3
|
|
|
|
___udivsi3:
|
|
|
|
PUSHP S2P
|
|
|
|
PUSHP S0P
|
|
|
|
PUSHP S1P
|
|
|
|
bsr divmodsi4
|
2003-11-13 23:28:07 +01:00
|
|
|
bra reti
|
2003-11-13 20:42:03 +01:00
|
|
|
#endif
|
1994-07-12 00:45:37 +02:00
|
|
|
|
|
|
|
.global ___umodsi3
|
|
|
|
___umodsi3:
|
2003-11-13 20:42:03 +01:00
|
|
|
#ifdef __H8300__
|
1994-07-12 00:45:37 +02:00
|
|
|
PUSHP S2P
|
|
|
|
PUSHP S0P
|
|
|
|
PUSHP S1P
|
|
|
|
bsr divmodsi4
|
|
|
|
mov S0,A0
|
|
|
|
mov S1,A1
|
2003-11-13 23:28:07 +01:00
|
|
|
bra reti
|
1994-07-12 00:45:37 +02:00
|
|
|
#else
|
2003-11-14 05:35:46 +01:00
|
|
|
bsr ___udivsi3
|
2003-11-14 03:06:37 +01:00
|
|
|
mov.l er3,er0
|
2003-11-13 20:42:03 +01:00
|
|
|
rts
|
1994-07-12 00:45:37 +02:00
|
|
|
#endif
|
2002-05-03 14:28:41 +02:00
|
|
|
|
1994-07-12 00:45:37 +02:00
|
|
|
.global ___divsi3
|
|
|
|
___divsi3:
|
2003-11-14 13:39:27 +01:00
|
|
|
#ifdef __H8300__
|
1994-07-12 00:45:37 +02:00
|
|
|
PUSHP S2P
|
|
|
|
PUSHP S0P
|
|
|
|
PUSHP S1P
|
|
|
|
jsr divnorm
|
|
|
|
jsr divmodsi4
|
2003-11-14 05:35:46 +01:00
|
|
|
#else
|
2003-11-14 13:39:27 +01:00
|
|
|
PUSHP S2P
|
|
|
|
jsr divnorm
|
2003-11-14 05:35:46 +01:00
|
|
|
bsr ___udivsi3
|
|
|
|
#endif
|
1994-07-12 00:45:37 +02:00
|
|
|
|
|
|
|
; examine what the sign should be
|
|
|
|
exitdiv:
|
2003-11-15 22:36:49 +01:00
|
|
|
btst #3,S2L
|
1994-07-12 00:45:37 +02:00
|
|
|
beq reti
|
2002-05-03 14:28:41 +02:00
|
|
|
|
1994-07-12 00:45:37 +02:00
|
|
|
; should be -ve
|
|
|
|
#ifdef __H8300__
|
|
|
|
not A0H
|
|
|
|
not A1H
|
|
|
|
not A0L
|
|
|
|
not A1L
|
|
|
|
|
|
|
|
add #1,A1L
|
|
|
|
addx #0,A1H
|
|
|
|
addx #0,A0L
|
1997-04-30 00:12:25 +02:00
|
|
|
addx #0,A0H
|
1994-07-12 00:45:37 +02:00
|
|
|
#else /* __H8300H__ */
|
|
|
|
neg.l A0P
|
|
|
|
#endif
|
|
|
|
|
|
|
|
reti:
|
2003-11-14 13:39:27 +01:00
|
|
|
#ifdef __H8300__
|
2003-11-13 23:28:07 +01:00
|
|
|
POPP S1P
|
|
|
|
POPP S0P
|
2003-11-14 13:39:27 +01:00
|
|
|
#endif
|
1994-07-12 00:45:37 +02:00
|
|
|
POPP S2P
|
2002-05-03 14:28:41 +02:00
|
|
|
rts
|
1994-07-12 00:45:37 +02:00
|
|
|
|
2002-11-02 13:51:05 +01:00
|
|
|
; takes A0/A1 numerator (A0P for H8/300H)
|
|
|
|
; A2/A3 denominator (A1P for H8/300H)
|
|
|
|
; returns A0/A1 quotient (A0P for H8/300H)
|
|
|
|
; S0/S1 remainder (S0P for H8/300H)
|
2003-11-13 20:46:01 +01:00
|
|
|
; trashes S2H
|
1994-07-12 00:45:37 +02:00
|
|
|
|
|
|
|
#ifdef __H8300__
|
|
|
|
|
|
|
|
divmodsi4:
|
|
|
|
sub.w S0,S0 ; zero play area
|
|
|
|
mov.w S0,S1
|
|
|
|
mov.b A2H,S2H
|
|
|
|
or A2L,S2H
|
|
|
|
or A3H,S2H
|
2003-11-11 19:36:39 +01:00
|
|
|
bne DenHighNonZero
|
1994-07-12 00:45:37 +02:00
|
|
|
mov.b A0H,A0H
|
|
|
|
bne NumByte0Zero
|
|
|
|
mov.b A0L,A0L
|
|
|
|
bne NumByte1Zero
|
|
|
|
mov.b A1H,A1H
|
|
|
|
bne NumByte2Zero
|
|
|
|
bra NumByte3Zero
|
|
|
|
NumByte0Zero:
|
|
|
|
mov.b A0H,S1L
|
|
|
|
divxu A3L,S1
|
|
|
|
mov.b S1L,A0H
|
|
|
|
NumByte1Zero:
|
|
|
|
mov.b A0L,S1L
|
|
|
|
divxu A3L,S1
|
|
|
|
mov.b S1L,A0L
|
|
|
|
NumByte2Zero:
|
|
|
|
mov.b A1H,S1L
|
|
|
|
divxu A3L,S1
|
|
|
|
mov.b S1L,A1H
|
|
|
|
NumByte3Zero:
|
|
|
|
mov.b A1L,S1L
|
|
|
|
divxu A3L,S1
|
|
|
|
mov.b S1L,A1L
|
|
|
|
|
|
|
|
mov.b S1H,S1L
|
|
|
|
mov.b #0x0,S1H
|
2002-05-03 14:28:41 +02:00
|
|
|
rts
|
1994-07-12 00:45:37 +02:00
|
|
|
|
|
|
|
; have to do the divide by shift and test
|
2003-11-11 19:36:39 +01:00
|
|
|
DenHighNonZero:
|
1994-07-12 00:45:37 +02:00
|
|
|
mov.b A0H,S1L
|
|
|
|
mov.b A0L,A0H
|
|
|
|
mov.b A1H,A0L
|
|
|
|
mov.b A1L,A1H
|
|
|
|
|
|
|
|
mov.b #0,A1L
|
|
|
|
mov.b #24,S2H ; only do 24 iterations
|
|
|
|
|
|
|
|
nextbit:
|
|
|
|
add.w A1,A1 ; double the answer guess
|
|
|
|
rotxl A0L
|
|
|
|
rotxl A0H
|
|
|
|
|
|
|
|
rotxl S1L ; double remainder
|
|
|
|
rotxl S1H
|
|
|
|
rotxl S0L
|
|
|
|
rotxl S0H
|
|
|
|
sub.w A3,S1 ; does it all fit
|
|
|
|
subx A2L,S0L
|
|
|
|
subx A2H,S0H
|
2002-05-03 14:28:41 +02:00
|
|
|
bhs setone
|
1994-07-12 00:45:37 +02:00
|
|
|
|
|
|
|
add.w A3,S1 ; no, restore mistake
|
|
|
|
addx A2L,S0L
|
|
|
|
addx A2H,S0H
|
|
|
|
|
|
|
|
dec S2H
|
|
|
|
bne nextbit
|
2002-05-03 14:28:41 +02:00
|
|
|
rts
|
|
|
|
|
1994-07-12 00:45:37 +02:00
|
|
|
setone:
|
|
|
|
inc A1L
|
|
|
|
dec S2H
|
|
|
|
bne nextbit
|
2002-05-03 14:28:41 +02:00
|
|
|
rts
|
1994-07-12 00:45:37 +02:00
|
|
|
|
|
|
|
#else /* __H8300H__ */
|
|
|
|
|
2003-11-14 13:49:40 +01:00
|
|
|
;; This function also computes the remainder and stores it in er3.
|
2003-11-14 05:35:46 +01:00
|
|
|
.global ___udivsi3
|
|
|
|
___udivsi3:
|
1994-07-12 00:45:37 +02:00
|
|
|
mov.w A1E,A1E ; denominator top word 0?
|
2003-11-11 19:36:39 +01:00
|
|
|
bne DenHighNonZero
|
1994-07-12 00:45:37 +02:00
|
|
|
|
|
|
|
; do it the easy way, see page 107 in manual
|
|
|
|
mov.w A0E,A2
|
|
|
|
extu.l A2P
|
|
|
|
divxu.w A1,A2P
|
|
|
|
mov.w A2E,A0E
|
|
|
|
divxu.w A1,A0P
|
2003-11-14 03:06:37 +01:00
|
|
|
mov.w A0E,A3
|
1994-07-12 00:45:37 +02:00
|
|
|
mov.w A2,A0E
|
2003-11-14 03:06:37 +01:00
|
|
|
extu.l A3P
|
1994-07-12 00:45:37 +02:00
|
|
|
rts
|
|
|
|
|
2003-11-13 18:10:23 +01:00
|
|
|
; er0 = er0 / er1
|
2003-11-14 03:06:37 +01:00
|
|
|
; er3 = er0 % er1
|
2003-11-13 18:10:23 +01:00
|
|
|
; trashes er1 er2
|
|
|
|
; expects er1 >= 2^16
|
2003-11-11 19:36:39 +01:00
|
|
|
DenHighNonZero:
|
2003-11-14 03:06:37 +01:00
|
|
|
mov.l er0,er3
|
2003-11-13 18:10:23 +01:00
|
|
|
mov.l er1,er2
|
|
|
|
#ifdef __H8300H__
|
|
|
|
divmod_L21:
|
|
|
|
shlr.l er0
|
|
|
|
shlr.l er2 ; make divisor < 2^16
|
|
|
|
mov.w e2,e2
|
|
|
|
bne divmod_L21
|
|
|
|
#else
|
2003-11-14 21:39:30 +01:00
|
|
|
shlr.l #2,er2 ; make divisor < 2^16
|
|
|
|
mov.w e2,e2
|
|
|
|
beq divmod_L22A
|
2003-11-13 18:10:23 +01:00
|
|
|
divmod_L21:
|
|
|
|
shlr.l #2,er0
|
|
|
|
divmod_L22:
|
|
|
|
shlr.l #2,er2 ; make divisor < 2^16
|
|
|
|
mov.w e2,e2
|
|
|
|
bne divmod_L21
|
2003-11-14 21:39:30 +01:00
|
|
|
divmod_L22A:
|
2003-11-13 18:10:23 +01:00
|
|
|
rotxl.w r2
|
|
|
|
bcs divmod_L23
|
|
|
|
shlr.l er0
|
|
|
|
bra divmod_L24
|
|
|
|
divmod_L23:
|
|
|
|
rotxr.w r2
|
|
|
|
shlr.l #2,er0
|
|
|
|
divmod_L24:
|
|
|
|
#endif
|
|
|
|
;; At this point,
|
|
|
|
;; er0 contains shifted dividend
|
|
|
|
;; er1 contains divisor
|
|
|
|
;; er2 contains shifted divisor
|
2003-11-14 03:06:37 +01:00
|
|
|
;; er3 contains dividend, later remainder
|
2003-11-13 20:46:01 +01:00
|
|
|
divxu.w r2,er0 ; r0 now contains the approximate quotient (AQ)
|
2003-11-13 18:10:23 +01:00
|
|
|
extu.l er0
|
|
|
|
beq divmod_L25
|
|
|
|
subs #1,er0 ; er0 = AQ - 1
|
|
|
|
mov.w e1,r2
|
|
|
|
mulxu.w r0,er2 ; er2 = upper (AQ - 1) * divisor
|
2003-11-14 03:06:37 +01:00
|
|
|
sub.w r2,e3 ; dividend - 65536 * er2
|
2003-11-13 18:10:23 +01:00
|
|
|
mov.w r1,r2
|
2003-11-14 03:06:37 +01:00
|
|
|
mulxu.w r0,er2 ; compute er3 = remainder (tentative)
|
|
|
|
sub.l er2,er3 ; er3 = dividend - (AQ - 1) * divisor
|
2003-11-13 18:10:23 +01:00
|
|
|
divmod_L25:
|
2003-11-14 03:06:37 +01:00
|
|
|
cmp.l er1,er3 ; is divisor < remainder?
|
2003-11-13 18:10:23 +01:00
|
|
|
blo divmod_L26
|
|
|
|
adds #1,er0
|
2003-11-14 03:06:37 +01:00
|
|
|
sub.l er1,er3 ; correct the remainder
|
2003-11-13 18:10:23 +01:00
|
|
|
divmod_L26:
|
1994-07-12 00:45:37 +02:00
|
|
|
rts
|
|
|
|
|
|
|
|
#endif
|
|
|
|
#endif /* L_divsi3 */
|
|
|
|
|
|
|
|
#ifdef L_mulhi3
|
|
|
|
|
|
|
|
;; HImode multiply.
|
2001-08-29 16:45:35 +02:00
|
|
|
; The H8/300 only has an 8*8->16 multiply.
|
1994-07-12 00:45:37 +02:00
|
|
|
; The answer is the same as:
|
2002-05-03 14:28:41 +02:00
|
|
|
;
|
1994-07-12 00:45:37 +02:00
|
|
|
; product = (srca.l * srcb.l) + ((srca.h * srcb.l) + (srcb.h * srca.l)) * 256
|
|
|
|
; (we can ignore A1.h * A0.h cause that will all off the top)
|
|
|
|
; A0 in
|
2002-05-03 14:28:41 +02:00
|
|
|
; A1 in
|
1994-07-12 00:45:37 +02:00
|
|
|
; A0 answer
|
|
|
|
|
|
|
|
#ifdef __H8300__
|
|
|
|
.section .text
|
|
|
|
.align 2
|
|
|
|
.global ___mulhi3
|
|
|
|
___mulhi3:
|
|
|
|
mov.b A1L,A2L ; A2l gets srcb.l
|
2002-05-03 14:28:41 +02:00
|
|
|
mulxu A0L,A2 ; A2 gets first sub product
|
1994-07-12 00:45:37 +02:00
|
|
|
|
|
|
|
mov.b A0H,A3L ; prepare for
|
|
|
|
mulxu A1L,A3 ; second sub product
|
|
|
|
|
|
|
|
add.b A3L,A2H ; sum first two terms
|
|
|
|
|
|
|
|
mov.b A1H,A3L ; third sub product
|
2002-05-03 14:28:41 +02:00
|
|
|
mulxu A0L,A3
|
1994-07-12 00:45:37 +02:00
|
|
|
|
|
|
|
add.b A3L,A2H ; almost there
|
|
|
|
mov.w A2,A0 ; that is
|
|
|
|
rts
|
|
|
|
|
|
|
|
#endif
|
|
|
|
#endif /* L_mulhi3 */
|
|
|
|
|
|
|
|
#ifdef L_mulsi3
|
|
|
|
|
|
|
|
;; SImode multiply.
|
2002-05-03 14:28:41 +02:00
|
|
|
;;
|
1994-07-12 00:45:37 +02:00
|
|
|
;; I think that shift and add may be sufficient for this. Using the
|
|
|
|
;; supplied 8x8->16 would need 10 ops of 14 cycles each + overhead. This way
|
|
|
|
;; the inner loop uses maybe 20 cycles + overhead, but terminates
|
|
|
|
;; quickly on small args.
|
|
|
|
;;
|
|
|
|
;; A0/A1 src_a
|
|
|
|
;; A2/A3 src_b
|
|
|
|
;;
|
2002-05-03 14:28:41 +02:00
|
|
|
;; while (a)
|
1994-07-12 00:45:37 +02:00
|
|
|
;; {
|
|
|
|
;; if (a & 1)
|
|
|
|
;; r += b;
|
|
|
|
;; a >>= 1;
|
|
|
|
;; b <<= 1;
|
|
|
|
;; }
|
|
|
|
|
|
|
|
.section .text
|
|
|
|
.align 2
|
|
|
|
|
|
|
|
#ifdef __H8300__
|
|
|
|
|
|
|
|
.global ___mulsi3
|
|
|
|
___mulsi3:
|
|
|
|
PUSHP S0P
|
|
|
|
PUSHP S1P
|
2002-05-03 14:28:41 +02:00
|
|
|
|
1994-07-12 00:45:37 +02:00
|
|
|
sub.w S0,S0
|
|
|
|
sub.w S1,S1
|
2002-05-03 14:28:41 +02:00
|
|
|
|
1994-07-12 00:45:37 +02:00
|
|
|
; while (a)
|
|
|
|
_top: mov.w A0,A0
|
|
|
|
bne _more
|
|
|
|
mov.w A1,A1
|
|
|
|
beq _done
|
|
|
|
_more: ; if (a & 1)
|
|
|
|
bld #0,A1L
|
|
|
|
bcc _nobit
|
|
|
|
; r += b
|
|
|
|
add.w A3,S1
|
|
|
|
addx A2L,S0L
|
|
|
|
addx A2H,S0H
|
|
|
|
_nobit:
|
|
|
|
; a >>= 1
|
|
|
|
shlr A0H
|
|
|
|
rotxr A0L
|
|
|
|
rotxr A1H
|
|
|
|
rotxr A1L
|
2002-05-03 14:28:41 +02:00
|
|
|
|
1994-07-12 00:45:37 +02:00
|
|
|
; b <<= 1
|
|
|
|
add.w A3,A3
|
|
|
|
addx A2L,A2L
|
|
|
|
addx A2H,A2H
|
|
|
|
bra _top
|
|
|
|
|
|
|
|
_done:
|
2002-05-03 14:28:41 +02:00
|
|
|
mov.w S0,A0
|
1994-07-12 00:45:37 +02:00
|
|
|
mov.w S1,A1
|
|
|
|
POPP S1P
|
|
|
|
POPP S0P
|
|
|
|
rts
|
|
|
|
|
|
|
|
#else /* __H8300H__ */
|
|
|
|
|
2002-04-20 20:01:53 +02:00
|
|
|
;
|
2004-01-15 20:49:41 +01:00
|
|
|
; mulsi3 for H8/300H - based on Renesas SH implementation
|
2002-04-20 20:01:53 +02:00
|
|
|
;
|
|
|
|
; by Toshiyasu Morita
|
|
|
|
;
|
|
|
|
; Old code:
|
|
|
|
;
|
|
|
|
; 16b * 16b = 372 states (worst case)
|
|
|
|
; 32b * 32b = 724 states (worst case)
|
|
|
|
;
|
|
|
|
; New code:
|
|
|
|
;
|
2002-04-20 22:21:35 +02:00
|
|
|
; 16b * 16b = 48 states
|
|
|
|
; 16b * 32b = 72 states
|
|
|
|
; 32b * 32b = 92 states
|
2002-04-20 20:01:53 +02:00
|
|
|
;
|
|
|
|
|
1994-07-12 00:45:37 +02:00
|
|
|
.global ___mulsi3
|
|
|
|
___mulsi3:
|
2002-04-20 20:01:53 +02:00
|
|
|
mov.w r1,r2 ; ( 2 states) b * d
|
|
|
|
mulxu r0,er2 ; (22 states)
|
2002-05-03 14:28:41 +02:00
|
|
|
|
2002-04-20 20:01:53 +02:00
|
|
|
mov.w e0,r3 ; ( 2 states) a * d
|
|
|
|
beq L_skip1 ; ( 4 states)
|
|
|
|
mulxu r1,er3 ; (22 states)
|
2002-04-20 22:21:35 +02:00
|
|
|
add.w r3,e2 ; ( 2 states)
|
2002-04-20 20:01:53 +02:00
|
|
|
|
|
|
|
L_skip1:
|
|
|
|
mov.w e1,r3 ; ( 2 states) c * b
|
|
|
|
beq L_skip2 ; ( 4 states)
|
|
|
|
mulxu r0,er3 ; (22 states)
|
2002-04-20 22:21:35 +02:00
|
|
|
add.w r3,e2 ; ( 2 states)
|
2002-04-20 20:01:53 +02:00
|
|
|
|
|
|
|
L_skip2:
|
|
|
|
mov.l er2,er0 ; ( 2 states)
|
|
|
|
rts ; (10 states)
|
1994-07-12 00:45:37 +02:00
|
|
|
|
|
|
|
#endif
|
1994-07-16 22:45:33 +02:00
|
|
|
#endif /* L_mulsi3 */
|
2004-03-03 05:53:16 +01:00
|
|
|
#ifdef L_fixunssfsi_asm
|
|
|
|
/* For the h8300 we use asm to save some bytes, to
|
|
|
|
allow more programs to fit into the tiny address
|
|
|
|
space. For the H8/300H and H8S, the C version is good enough. */
|
|
|
|
#ifdef __H8300__
|
|
|
|
/* We still treat NANs different than libgcc2.c, but then, the
|
|
|
|
behavior is undefined anyways. */
|
|
|
|
.global ___fixunssfsi
|
|
|
|
___fixunssfsi:
|
|
|
|
cmp.b #0x4f,r0h
|
|
|
|
bge Large_num
|
|
|
|
jmp @___fixsfsi
|
|
|
|
Large_num:
|
|
|
|
bhi L_huge_num
|
|
|
|
xor.b #0x80,A0L
|
|
|
|
bmi L_shift8
|
|
|
|
L_huge_num:
|
|
|
|
mov.w #65535,A0
|
|
|
|
mov.w A0,A1
|
|
|
|
rts
|
|
|
|
L_shift8:
|
|
|
|
mov.b A0L,A0H
|
|
|
|
mov.b A1H,A0L
|
|
|
|
mov.b A1L,A1H
|
|
|
|
mov.b #0,A1L
|
|
|
|
rts
|
|
|
|
#endif
|
|
|
|
#endif /* L_fixunssfsi_asm */
|