+2017-09-15 Richard Sandiford <richard.sandiford@linaro.org>
+ Alan Hayard <alan.hayward@arm.com>
+ David Sherwood <david.sherwood@arm.com>
+
+ * target.def (truly_noop_truncation): New hook.
+ (mode_rep_extended): Refer to TARGET_TRULY_NOOP_TRUNCATION rather
+ than TRULY_NOOP_TRUNCATION.
+ * hooks.h (hook_bool_uint_uint_true): Declare.
+ * hooks.c (hook_bool_uint_uint_true): New function.
+ * doc/tm.texi.in (TRULY_NOOP_TRUNCATION): Replace with...
+ (TARGET_TRULY_NOOP_TRUNCATION): ...this.
+ * doc/tm.texi: Regenerate.
+ * combine.c (make_extraction): Refer to TARGET_TRULY_NOOP_TRUNCATION
+ rather than TRULY_NOOP_TRUNCATION in comments.
+ (simplify_comparison): Likewise.
+ (record_truncated_value): Likewise.
+ * expmed.c (extract_bit_field_1): Likewise.
+ (extract_split_bit_field): Likewise.
+ * convert.c (convert_to_integer_1): Use targetm.truly_noop_truncation
+ instead of TRULY_NOOP_TRUNCATION.
+ * function.c (assign_parm_setup_block): Likewise.
+ * machmode.h (TRULY_NOOP_TRUNCATION_MODES_P): Likewise.
+ * rtlhooks.c: Include target.h.
+ * config/aarch64/aarch64.h (TRULY_NOOP_TRUNCATION): Delete.
+ * config/alpha/alpha.h (TRULY_NOOP_TRUNCATION): Delete.
+ * config/arc/arc.h (TRULY_NOOP_TRUNCATION): Delete.
+ * config/arm/arm.h (TRULY_NOOP_TRUNCATION): Delete.
+ * config/avr/avr.h (TRULY_NOOP_TRUNCATION): Delete.
+ * config/bfin/bfin.h (TRULY_NOOP_TRUNCATION): Delete.
+ * config/c6x/c6x.h (TRULY_NOOP_TRUNCATION): Delete.
+ * config/cr16/cr16.h (TRULY_NOOP_TRUNCATION): Delete.
+ * config/cris/cris.h (TRULY_NOOP_TRUNCATION): Delete.
+ * config/epiphany/epiphany.h (TRULY_NOOP_TRUNCATION): Delete.
+ * config/fr30/fr30.h (TRULY_NOOP_TRUNCATION): Delete.
+ * config/frv/frv.h (TRULY_NOOP_TRUNCATION): Delete.
+ * config/ft32/ft32.h (TRULY_NOOP_TRUNCATION): Delete.
+ * config/h8300/h8300.h (TRULY_NOOP_TRUNCATION): Delete.
+ * config/i386/i386.h (TRULY_NOOP_TRUNCATION): Delete.
+ * config/ia64/ia64.h (TRULY_NOOP_TRUNCATION): Delete.
+ * config/iq2000/iq2000.h (TRULY_NOOP_TRUNCATION): Delete.
+ * config/lm32/lm32.h (TRULY_NOOP_TRUNCATION): Delete.
+ * config/m32c/m32c.h (TRULY_NOOP_TRUNCATION): Delete.
+ * config/m32r/m32r.h (TRULY_NOOP_TRUNCATION): Delete.
+ * config/m68k/m68k.h (TRULY_NOOP_TRUNCATION): Delete.
+ * config/mcore/mcore.h (TRULY_NOOP_TRUNCATION): Delete.
+ * config/microblaze/microblaze.h (TRULY_NOOP_TRUNCATION): Delete.
+ * config/mips/mips.h (TRULY_NOOP_TRUNCATION): Delete.
+ * config/mips/mips.c (mips_truly_noop_truncation): New function.
+ (TARGET_TRULY_NOOP_TRUNCATION): Redefine.
+ * config/mips/mips.md: Refer to TARGET_TRULY_NOOP_TRUNCATION
+ rather than TRULY_NOOP_TRUNCATION in comments.
+ * config/mmix/mmix.h (TRULY_NOOP_TRUNCATION): Delete.
+ * config/mn10300/mn10300.h (TRULY_NOOP_TRUNCATION): Delete.
+ * config/moxie/moxie.h (TRULY_NOOP_TRUNCATION): Delete.
+ * config/msp430/msp430.h (TRULY_NOOP_TRUNCATION): Delete.
+ * config/nds32/nds32.h (TRULY_NOOP_TRUNCATION): Delete.
+ * config/nios2/nios2.h (TRULY_NOOP_TRUNCATION): Delete.
+ * config/nvptx/nvptx.h (TRULY_NOOP_TRUNCATION): Delete.
+ * config/pa/pa.h (TRULY_NOOP_TRUNCATION): Delete.
+ * config/pdp11/pdp11.h (TRULY_NOOP_TRUNCATION): Delete.
+ * config/powerpcspe/powerpcspe.h (TRULY_NOOP_TRUNCATION): Delete.
+ * config/riscv/riscv.h (TRULY_NOOP_TRUNCATION): Delete.
+ * config/riscv/riscv.md: Refer to TARGET_TRULY_NOOP_TRUNCATION
+ rather than TRULY_NOOP_TRUNCATION in comments.
+ * config/rl78/rl78.h (TRULY_NOOP_TRUNCATION): Delete.
+ * config/rs6000/rs6000.h (TRULY_NOOP_TRUNCATION): Delete.
+ * config/rx/rx.h (TRULY_NOOP_TRUNCATION): Delete.
+ * config/s390/s390.h (TRULY_NOOP_TRUNCATION): Delete.
+ * config/sh/sh.h (MAYBE_BASE_REGISTER_RTX_P): Remove
+ TRULY_NOOP_TRUNCATION condition.
+ (MAYBE_INDEX_REGISTER_RTX_P): Likewise.
+ (TRULY_NOOP_TRUNCATION): Delete.
+ * config/sparc/sparc.h (TRULY_NOOP_TRUNCATION): Delete.
+ * config/spu/spu.h (TRULY_NOOP_TRUNCATION): Delete.
+ * config/spu/spu.c (spu_truly_noop_truncation): New function.
+ (TARGET_TRULY_NOOP_TRUNCATION): Redefine.
+ * config/stormy16/stormy16.h (TRULY_NOOP_TRUNCATION): Delete.
+ * config/tilegx/tilegx.h (TRULY_NOOP_TRUNCATION): Delete.
+ * config/tilegx/tilegx.c (tilegx_truly_noop_truncation): New fuction.
+ (TARGET_TRULY_NOOP_TRUNCATION): Redefine.
+ * config/tilegx/tilegx.md: Refer to TARGET_TRULY_NOOP_TRUNCATION
+ rather than TRULY_NOOP_TRUNCATION in comments.
+ * config/tilepro/tilepro.h (TRULY_NOOP_TRUNCATION): Delete.
+ * config/v850/v850.h (TRULY_NOOP_TRUNCATION): Delete.
+ * config/vax/vax.h (TRULY_NOOP_TRUNCATION): Delete.
+ * config/visium/visium.h (TRULY_NOOP_TRUNCATION): Delete.
+ * config/xtensa/xtensa.h (TRULY_NOOP_TRUNCATION): Delete.
+ * system.h (TRULY_NOOP_TRUNCATION): Poison.
+
2017-09-15 Christophe Lyon <christophe.lyon@linaro.org>
PR target/67591
else if (!MEM_P (inner))
{
/* On the LHS, don't create paradoxical subregs implicitely truncating
- the register unless TRULY_NOOP_TRUNCATION. */
+ the register unless TARGET_TRULY_NOOP_TRUNCATION. */
if (in_dest
&& !TRULY_NOOP_TRUNCATION_MODES_P (GET_MODE (inner),
wanted_inner_mode))
(ne:DI (and:DI (reg:DI 4) (const_int 0xffffffff)) (const_int 0))
-> (ne:DI (reg:SI 4) (const_int 0))
- unless TRULY_NOOP_TRUNCATION allows it or the register is
+ unless TARGET_TRULY_NOOP_TRUNCATION allows it or the register is
known to hold a value of the required mode the
transformation is invalid. */
if ((equality_comparison_p || unsigned_comparison_p)
}
/* If X is a hard reg or a subreg record the mode that the register is
- accessed in. For non-TRULY_NOOP_TRUNCATION targets we might be able
- to turn a truncate into a subreg using this information. Return true
+ accessed in. For non-TARGET_TRULY_NOOP_TRUNCATION targets we might be
+ able to turn a truncate into a subreg using this information. Return true
if traversing X is complete. */
static bool
if we don't have to, for power-saving reasons. */
#define SLOW_BYTE_ACCESS 0
-#define TRULY_NOOP_TRUNCATION(OUTPREC, INPREC) 1
-
#define NO_FUNCTION_CSE 1
/* Specify the machine mode that the hardware addresses have.
/* Define if loading short immediate values into registers sign extends. */
#define SHORT_IMMEDIATES_SIGN_EXTEND 1
-/* Value is 1 if truncating an integer of INPREC bits to OUTPREC bits
- is done just by pretending it is already truncated. */
-#define TRULY_NOOP_TRUNCATION(OUTPREC, INPREC) 1
-
/* The CIX ctlz and cttz instructions return 64 for zero. */
#define CLZ_DEFINED_VALUE_AT_ZERO(MODE, VALUE) ((VALUE) = 64, \
TARGET_CIX ? 1 : 0)
*/
#define SHIFT_COUNT_TRUNCATED 1
-/* Value is 1 if truncating an integer of INPREC bits to OUTPREC bits
- is done just by pretending it is already truncated. */
-#define TRULY_NOOP_TRUNCATION(OUTPREC, INPREC) 1
-
/* We assume that the store-condition-codes instructions store 0 for false
and some other value for true. This is the value stored for true. */
#define STORE_FLAG_VALUE 1
rotates is modulo 32 used. */
/* #define SHIFT_COUNT_TRUNCATED 1 */
-/* All integers have the same format so truncation is easy. */
-#define TRULY_NOOP_TRUNCATION(OUTPREC, INPREC) 1
-
/* Calling from registers is a massive pain. */
#define NO_FUNCTION_CSE 1
#define MOVE_RATIO(speed) ((speed) ? 3 : 2)
-#define TRULY_NOOP_TRUNCATION(OUTPREC, INPREC) 1
-
#define Pmode HImode
#define FUNCTION_MODE HImode
#define NOTICE_UPDATE_CC(EXPR, INSN) 0
-/* Value is 1 if truncating an integer of INPREC bits to OUTPREC bits
- is done just by pretending it is already truncated. */
-#define TRULY_NOOP_TRUNCATION(OUTPREC, INPREC) 1
-
/* Max number of bytes we can move from memory to memory
in one reasonably fast instruction. */
#define MOVE_MAX UNITS_PER_WORD
#define CASE_VECTOR_MODE SImode
#define MOVE_MAX 4
#define MOVE_RATIO(SPEED) 4
-#define TRULY_NOOP_TRUNCATION(outprec, inprec) 1
#define CLZ_DEFINED_VALUE_AT_ZERO(MODE, VALUE) ((VALUE) = 32, 1)
#define Pmode SImode
#define FUNCTION_MODE QImode
#define MOVE_MAX 4
-#define TRULY_NOOP_TRUNCATION(OUTPREC, INPREC) 1
-
#define STORE_FLAG_VALUE 1
#define Pmode SImode
/* Maybe SHIFT_COUNT_TRUNCATED is safe to define? FIXME: Check later. */
-#define TRULY_NOOP_TRUNCATION(OUTPREC, INPREC) 1
-
#define CLZ_DEFINED_VALUE_AT_ZERO(MODE, VALUE) ((VALUE) = 32, 1)
#define CTZ_DEFINED_VALUE_AT_ZERO(MODE, VALUE) ((VALUE) = 32, 1)
few bits. */
#define SHIFT_COUNT_TRUNCATED 1
-/* Value is 1 if truncating an integer of INPREC bits to OUTPREC bits
- is done just by pretending it is already truncated. */
-#define TRULY_NOOP_TRUNCATION(OUTPREC, INPREC) 1
-
/* Specify the machine mode that pointers have.
After generation of rtl, the compiler makes no further distinction
between pointers and any other objects of this machine mode. */
memory to memory. */
#define MOVE_MAX 8
-/* A C expression which is nonzero if on this machine it is safe to "convert"
- an integer of INPREC bits to one of OUTPREC bits (where OUTPREC is smaller
- than INPREC) by merely operating on it as if it had only OUTPREC bits.
-
- On many machines, this expression can be 1.
-
- When `TRULY_NOOP_TRUNCATION' returns 1 for a pair of sizes for modes for
- which `TARGET_MODES_TIEABLE_P' is 0, suboptimal code can result.
- If this is the case, making `TRULY_NOOP_TRUNCATION' return 0 in such
- cases may improve things. */
-#define TRULY_NOOP_TRUNCATION(OUTPREC, INPREC) 1
-
/* An alias for the machine mode for pointers. On most machines, define this
to be the integer mode corresponding to the width of a hardware pointer;
`SImode' on 32-bit machine or `DImode' on 64-bit machines. On some machines
memory to memory. */
#define MOVE_MAX 8
-/* A C expression which is nonzero if on this machine it is safe to "convert"
- an integer of INPREC bits to one of OUTPREC bits (where OUTPREC is smaller
- than INPREC) by merely operating on it as if it had only OUTPREC bits.
-
- On many machines, this expression can be 1.
-
- When `TRULY_NOOP_TRUNCATION' returns 1 for a pair of sizes for modes
- for which `TARGET_MODES_TIEABLE_P' is 0, suboptimal code can result.
- If this is the case, making `TRULY_NOOP_TRUNCATION' return 0 in such
- cases may improve things. */
-#define TRULY_NOOP_TRUNCATION(OUTPREC, INPREC) 1
-
/* An alias for the machine mode for pointers. On most machines, define this
to be the integer mode corresponding to the width of a hardware pointer;
`SImode' on 32-bit machine or `DImode' on 64-bit machines. On some machines
quickly between memory and registers or between two memory
locations. */
#define MOVE_MAX 4
-#define TRULY_NOOP_TRUNCATION(op,ip) 1
/* Define this to be nonzero if shift instructions ignore all but the low-order
few bits. */
of a shift count. */
/* #define SHIFT_COUNT_TRUNCATED */
-/* Value is 1 if truncating an integer of INPREC bits to OUTPREC bits
- is done just by pretending it is already truncated. */
-#define TRULY_NOOP_TRUNCATION(OUTPREC, INPREC) 1
-
/* Specify the machine mode that pointers have.
After generation of rtl, the compiler makes no further distinction
between pointers and any other objects of this machine mode. */
/* #define SHIFT_COUNT_TRUNCATED */
-/* Value is 1 if truncating an integer of INPREC bits to OUTPREC bits
- is done just by pretending it is already truncated. */
-#define TRULY_NOOP_TRUNCATION(OUTPREC, INPREC) 1
-
/* A macro to update M and UNSIGNEDP when an object whose type is
TYPE and which has the specified mode and signedness is to be
stored in a register. This macro is only called when TYPE is a
memory to memory. */
#define MOVE_MAX 8
-/* A C expression which is nonzero if on this machine it is safe to "convert"
- an integer of INPREC bits to one of OUTPREC bits (where OUTPREC is smaller
- than INPREC) by merely operating on it as if it had only OUTPREC bits. */
-
-#define TRULY_NOOP_TRUNCATION(OUTPREC, INPREC) 1
-
/* A C expression describing the value returned by a comparison operator with
an integral mode and stored by a store-flag instruction (`sCOND') when the
condition is true. */
#define SHIFT_COUNT_TRUNCATED 1
-#define TRULY_NOOP_TRUNCATION(OUTPREC, INPREC) 1
-
#define STORE_FLAG_VALUE 1
#define Pmode SImode
#define SHIFT_COUNT_TRUNCATED 1
-#define TRULY_NOOP_TRUNCATION(OUTPREC, INPREC) 1
-
#define Pmode SImode
#define FUNCTION_MODE SImode
#define LOAD_EXTEND_OP(MEM) ZERO_EXTEND
#define MOVE_MAX 4
-#define TRULY_NOOP_TRUNCATION(op,ip) 1
#define STORE_FLAG_VALUE 1
few bits. */
#define SHIFT_COUNT_TRUNCATED 1
-/* Value is 1 if truncating an integer of INPREC bits to OUTPREC bits
- is done just by pretending it is already truncated. */
-#define TRULY_NOOP_TRUNCATION(OUTPREC, INPREC) 1
-
/* Specify the machine mode that pointers have.
After generation of rtl, the compiler makes no further distinction
between pointers and any other objects of this machine mode. */
#define MOVE_MAX 4
#define SLOW_BYTE_ACCESS 0
-#define TRULY_NOOP_TRUNCATION(OUTPREC, INPREC) 1
-
/* The 68020 BFFFO and ColdFire FF1 instructions return 32 for zero. */
#define CLZ_DEFINED_VALUE_AT_ZERO(MODE, VALUE) ((VALUE) = 32, 1)
target. */
#define SHIFT_COUNT_TRUNCATED 0
-/* All integers have the same format so truncation is easy. */
-#define TRULY_NOOP_TRUNCATION(OUTPREC,INPREC) 1
-
/* Define this if addresses of constant functions
shouldn't be put through pseudo regs where they can be cse'd.
Desirable on machines where ordinary constants are expensive
#define SHIFT_COUNT_TRUNCATED 1
-/* This results in inefficient code for 64 bit to 32 conversions.
- Something needs to be done about this. Perhaps not use any 32 bit
- instructions? Perhaps use PROMOTE_MODE? */
-#define TRULY_NOOP_TRUNCATION(OUTPREC, INPREC) 1
-
#define Pmode SImode
#define FUNCTION_MODE SImode
*punsignedp = unsignedp;
return mode;
}
+
+/* Implement TARGET_TRULY_NOOP_TRUNCATION. */
+
+static bool
+mips_truly_noop_truncation (unsigned int outprec, unsigned int inprec)
+{
+ return !TARGET_64BIT || inprec <= 32 || outprec > 32;
+}
\f
/* Initialize the GCC target structure. */
#undef TARGET_ASM_ALIGNED_HI_OP
#undef TARGET_CAN_CHANGE_MODE_CLASS
#define TARGET_CAN_CHANGE_MODE_CLASS mips_can_change_mode_class
+#undef TARGET_TRULY_NOOP_TRUNCATION
+#define TARGET_TRULY_NOOP_TRUNCATION mips_truly_noop_truncation
+
struct gcc_target targetm = TARGET_INITIALIZER;
\f
#include "gt-mips.h"
do not truncate the shift amount at all. */
#define SHIFT_COUNT_TRUNCATED (!TARGET_LOONGSON_VECTORS)
-/* Value is 1 if truncating an integer of INPREC bits to OUTPREC bits
- is done just by pretending it is already truncated. */
-#define TRULY_NOOP_TRUNCATION(OUTPREC, INPREC) \
- (TARGET_64BIT ? ((INPREC) <= 32 || (OUTPREC) > 32) : 1)
-
/* Specify the machine mode that pointers have.
After generation of rtl, the compiler makes no further distinction
(match_operand:GPR 2 "and_reg_operand")))])
;; The middle-end is not allowed to convert ANDing with 0xffff_ffff into a
-;; zero_extendsidi2 because of TRULY_NOOP_TRUNCATION, so handle these here.
-;; Note that this variant does not trigger for SI mode because we require
-;; a 64-bit HOST_WIDE_INT and 0xffff_ffff wouldn't be a canonical
+;; zero_extendsidi2 because of TARGET_TRULY_NOOP_TRUNCATION, so handle these
+;; here. Note that this variant does not trigger for SI mode because we
+;; require a 64-bit HOST_WIDE_INT and 0xffff_ffff wouldn't be a canonical
;; sign-extended SImode value.
;;
;; These are possible combinations for operand 1 and 2. The table
;; modes is a no-op, as it is for most other GCC ports. Truncating
;; DImode values to SImode is not a no-op for TARGET_64BIT since we
;; need to make sure that the lower 32 bits are properly sign-extended
-;; (see TRULY_NOOP_TRUNCATION). Truncating DImode values into modes
+;; (see TARGET_TRULY_NOOP_TRUNCATION). Truncating DImode values into modes
;; smaller than SImode is equivalent to two separate truncations:
;;
;; A B
;; Those for integer source operand are ordered widest source type first.
;; When TARGET_64BIT, all SImode integer and accumulator registers
-;; should already be in sign-extended form (see TRULY_NOOP_TRUNCATION
+;; should already be in sign-extended form (see TARGET_TRULY_NOOP_TRUNCATION
;; and truncdisi2). We can therefore get rid of register->register
;; instructions if we constrain the source to be in the same register as
;; the destination.
#define MOVE_MAX 8
-#define TRULY_NOOP_TRUNCATION(OUTPREC, INPREC) 1
-
/* ??? MMIX allows a choice of STORE_FLAG_VALUE. Revisit later,
we don't have scc expanders yet. */
of a shift count. */
#define SHIFT_COUNT_TRUNCATED 1
-/* Value is 1 if truncating an integer of INPREC bits to OUTPREC bits
- is done just by pretending it is already truncated. */
-#define TRULY_NOOP_TRUNCATION(OUTPREC, INPREC) 1
-
/* Specify the machine mode that pointers have.
After generation of rtl, the compiler makes no further distinction
between pointers and any other objects of this machine mode. */
quickly between memory and registers or between two memory
locations. */
#define MOVE_MAX 4
-#define TRULY_NOOP_TRUNCATION(op,ip) 1
/* All load operations zero extend. */
#define LOAD_EXTEND_OP(MEM) ZERO_EXTEND
valid memory address. */
#define MAX_REGS_PER_ADDRESS 1
-#define TRULY_NOOP_TRUNCATION(op,ip) 1
-
/* An alias for a machine mode name. This is the machine mode that
elements of a jump-table should have. */
#define CASE_VECTOR_MODE SImode
#define RETURN_ADDR_RTX(COUNT, FA) \
msp430_return_addr_rtx (COUNT)
-#define TRULY_NOOP_TRUNCATION(OUTPREC, INPREC) 1
-
#define SLOW_BYTE_ACCESS 0
\f
of bits needed to represent the size of the object being shifted. */
#define SHIFT_COUNT_TRUNCATED 1
-/* A C expression which is nonzero if on this machine it is safe to "convert"
- an integer of 'inprec' bits to one of 'outprec' bits by merely operating
- on it as if it had only 'outprec' bits. */
-#define TRULY_NOOP_TRUNCATION(outprec, inprec) 1
-
/* A C expression describing the value returned by a comparison operator with
an integral mode and stored by a store-flag instruction ('cstoremode4')
when the condition is true. */
#define CASE_VECTOR_MODE Pmode
-#define TRULY_NOOP_TRUNCATION(OUTPREC, INPREC) 1
-
#define LOAD_EXTEND_OP(MODE) (ZERO_EXTEND)
#define WORD_REGISTER_OPERATIONS 1
#define CASE_VECTOR_MODE SImode
#define MOVE_MAX 8
#define MOVE_RATIO(SPEED) 4
-#define TRULY_NOOP_TRUNCATION(outprec, inprec) 1
#define FUNCTION_MODE QImode
#define HAS_INIT_SECTION 1
/* Nonzero if access to memory by bytes is slow and undesirable. */
#define SLOW_BYTE_ACCESS 1
-/* Value is 1 if truncating an integer of INPREC bits to OUTPREC bits
- is done just by pretending it is already truncated. */
-#define TRULY_NOOP_TRUNCATION(OUTPREC, INPREC) 1
-
/* Specify the machine mode that pointers have.
After generation of rtl, the compiler makes no further distinction
between pointers and any other objects of this machine mode. */
/* Do not break .stabs pseudos into continuations. */
#define DBX_CONTIN_LENGTH 0
-/* Value is 1 if truncating an integer of INPREC bits to OUTPREC bits
- is done just by pretending it is already truncated. */
-#define TRULY_NOOP_TRUNCATION(OUTPREC, INPREC) 1
-
/* Give a comparison code (EQ, NE etc) and the first operand of a COMPARE,
return the mode to be used for the comparison. For floating-point, CCFPmode
should be used. */
/* Define if loading short immediate values into registers sign extends. */
#define SHORT_IMMEDIATES_SIGN_EXTEND 1
\f
-/* Value is 1 if truncating an integer of INPREC bits to OUTPREC bits
- is done just by pretending it is already truncated. */
-#define TRULY_NOOP_TRUNCATION(OUTPREC, INPREC) 1
-
/* The cntlzw and cntlzd instructions return 32 and 64 for input of zero. */
#define CLZ_DEFINED_VALUE_AT_ZERO(MODE, VALUE) \
((VALUE) = GET_MODE_BITSIZE (MODE), 2)
#define SHIFT_COUNT_TRUNCATED 1
-#define TRULY_NOOP_TRUNCATION(OUTPREC, INPREC) 1
-
/* Specify the machine mode that pointers have.
After generation of rtl, the compiler makes no further distinction
between pointers and any other objects of this machine mode. */
(set_attr "mode" "HI")])
;; HImode constant generation; see riscv_move_integer for details.
-;; si+si->hi without truncation is legal because of TRULY_NOOP_TRUNCATION.
+;; si+si->hi without truncation is legal because of
+;; TARGET_TRULY_NOOP_TRUNCATION.
(define_insn "*add<mode>hi3"
[(set (match_operand:HI 0 "register_operand" "=r,r")
#define MOVE_MAX 2
#define STARTING_FRAME_OFFSET 0
-#define TRULY_NOOP_TRUNCATION(OUTPREC, INPREC) 1
-
#define ADDR_SPACE_NEAR 1
#define ADDR_SPACE_FAR 2
/* Define if loading short immediate values into registers sign extends. */
#define SHORT_IMMEDIATES_SIGN_EXTEND 1
\f
-/* Value is 1 if truncating an integer of INPREC bits to OUTPREC bits
- is done just by pretending it is already truncated. */
-#define TRULY_NOOP_TRUNCATION(OUTPREC, INPREC) 1
-
/* The cntlzw and cntlzd instructions return 32 and 64 for input of zero. */
#define CLZ_DEFINED_VALUE_AT_ZERO(MODE, VALUE) \
((VALUE) = GET_MODE_BITSIZE (MODE), 2)
#define MOVE_MAX 4
#define STARTING_FRAME_OFFSET 0
-#define TRULY_NOOP_TRUNCATION(OUTPREC, INPREC) 1
-
#define HAVE_PRE_DECREMENT 1
#define HAVE_POST_INCREMENT 1
tablejump instruction. */
#define CASE_VECTOR_MODE (TARGET_64BIT ? DImode : SImode)
-/* Value is 1 if truncating an integer of INPREC bits to OUTPREC bits
- is done just by pretending it is already truncated. */
-#define TRULY_NOOP_TRUNCATION(OUTPREC, INPREC) 1
-
/* Specify the machine mode that pointers have.
After generation of rtl, the compiler makes no further distinction
between pointers and any other objects of this machine mode. */
#define MAYBE_BASE_REGISTER_RTX_P(X, STRICT) \
((REG_P (X) && REG_OK_FOR_BASE_P (X, STRICT)) \
|| (GET_CODE (X) == SUBREG \
- && TRULY_NOOP_TRUNCATION (GET_MODE_BITSIZE (GET_MODE ((X))), \
- GET_MODE_BITSIZE (GET_MODE (SUBREG_REG (X)))) \
&& REG_P (SUBREG_REG (X)) \
&& REG_OK_FOR_BASE_P (SUBREG_REG (X), STRICT)))
#define MAYBE_INDEX_REGISTER_RTX_P(X, STRICT) \
((REG_P (X) && REG_OK_FOR_INDEX_P (X, STRICT)) \
|| (GET_CODE (X) == SUBREG \
- && TRULY_NOOP_TRUNCATION (GET_MODE_BITSIZE (GET_MODE ((X))), \
- GET_MODE_BITSIZE (GET_MODE (SUBREG_REG (X)))) \
&& REG_P (SUBREG_REG (X)) \
&& SUBREG_OK_FOR_INDEX_P (SUBREG_REG (X), SUBREG_BYTE (X), STRICT)))
more compact code. */
#define SHIFT_COUNT_TRUNCATED (0)
-/* All integers have the same format so truncation is easy. */
-#define TRULY_NOOP_TRUNCATION(OUTPREC,INPREC) (true)
-
/* Define this if addresses of constant functions
shouldn't be put through pseudo regs where they can be cse'd.
Desirable on machines where ordinary constants are expensive
few bits. */
#define SHIFT_COUNT_TRUNCATED 1
-/* Value is 1 if truncating an integer of INPREC bits to OUTPREC bits
- is done just by pretending it is already truncated. */
-#define TRULY_NOOP_TRUNCATION(OUTPREC, INPREC) 1
-
/* For SImode, we make sure the top 32-bits of the register are clear and
then we subtract 32 from the lzd instruction result. */
#define CLZ_DEFINED_VALUE_AT_ZERO(MODE, VALUE) \
|| (GET_MODE_SIZE (from) <= 4 && GET_MODE_SIZE (to) <= 4)
|| (GET_MODE_SIZE (from) >= 16 && GET_MODE_SIZE (to) >= 16));
}
+
+/* Implement TARGET_TRULY_NOOP_TRUNCATION. */
+
+static bool
+spu_truly_noop_truncation (unsigned int outprec, unsigned int inprec)
+{
+ return inprec <= 32 && outprec <= inprec;
+}
\f
/* Table of machine attributes. */
static const struct attribute_spec spu_attribute_table[] =
#undef TARGET_CAN_CHANGE_MODE_CLASS
#define TARGET_CAN_CHANGE_MODE_CLASS spu_can_change_mode_class
+#undef TARGET_TRULY_NOOP_TRUNCATION
+#define TARGET_TRULY_NOOP_TRUNCATION spu_truly_noop_truncation
+
struct gcc_target targetm = TARGET_INITIALIZER;
#include "gt-spu.h"
#define MOVE_MAX 16
-#define TRULY_NOOP_TRUNCATION(OUTPREC, INPREC) ((INPREC) <= 32 && (OUTPREC) <= (INPREC))
-
#define STORE_FLAG_VALUE -1
#define Pmode SImode
#define SHIFT_COUNT_TRUNCATED 1
-#define TRULY_NOOP_TRUNCATION(OUTPREC, INPREC) 1
-
#define Pmode HImode
#define FUNCTION_MODE HImode
file_end_indicate_exec_stack ();
}
+/* Implement TARGET_TRULY_NOOP_TRUNCATION. We represent all SI values
+ as sign-extended DI values in registers. */
+static bool
+tilegx_truly_noop_truncation (unsigned int outprec, unsigned int inprec)
+{
+ return inprec <= 32 || outprec > 32;
+}
#undef TARGET_HAVE_TLS
#define TARGET_HAVE_TLS HAVE_AS_TLS
#undef TARGET_CAN_USE_DOLOOP_P
#define TARGET_CAN_USE_DOLOOP_P can_use_doloop_if_innermost
+#undef TARGET_TRULY_NOOP_TRUNCATION
+#define TARGET_TRULY_NOOP_TRUNCATION tilegx_truly_noop_truncation
+
struct gcc_target targetm = TARGET_INITIALIZER;
#include "gt-tilegx.h"
#define SHORT_IMMEDIATES_SIGN_EXTEND 1
-/* We represent all SI values as sign-extended DI values in
- registers. */
-#define TRULY_NOOP_TRUNCATION(OUTPREC, INPREC) \
- ((INPREC) <= 32 || (OUTPREC) > 32)
-
#define CLZ_DEFINED_VALUE_AT_ZERO(MODE, VALUE) ((VALUE) = 64, 1)
#define CTZ_DEFINED_VALUE_AT_ZERO(MODE, VALUE) ((VALUE) = 64, 1)
ld2s_add\t%0, %I1, %i1"
[(set_attr "type" "X0,Y2_2cycle,X1_2cycle")])
-;; All SImode integer registers should already be in sign-extended
-;; form (see TRULY_NOOP_TRUNCATION and truncdisi2). We can therefore
+;; All SImode integer registers should already be in sign-extended form
+;; (see TARGET_TRULY_NOOP_TRUNCATION and truncdisi2). We can therefore
;; get rid of register->register instructions if we constrain the
;; source to be in the same register as the destination.
(define_insn_and_split "extendsidi2"
;; modes is a no-op, as it is for most other GCC ports. Truncating
;; DImode values to SImode is not a no-op since we
;; need to make sure that the lower 32 bits are properly sign-extended
-;; (see TRULY_NOOP_TRUNCATION). Truncating DImode values into modes
+;; (see TARGET_TRULY_NOOP_TRUNCATION). Truncating DImode values into modes
;; smaller than SImode is equivalent to two separate truncations:
;;
;; A B
#define SHORT_IMMEDIATES_SIGN_EXTEND 1
-#define TRULY_NOOP_TRUNCATION(OUTPREC, INPREC) 1
-
#define CLZ_DEFINED_VALUE_AT_ZERO(MODE, VALUE) ((VALUE) = 32, 1)
#define CTZ_DEFINED_VALUE_AT_ZERO(MODE, VALUE) ((VALUE) = 32, 1)
of a shift count. */
#define SHIFT_COUNT_TRUNCATED 1
-/* Value is 1 if truncating an integer of INPREC bits to OUTPREC bits
- is done just by pretending it is already truncated. */
-#define TRULY_NOOP_TRUNCATION(OUTPREC, INPREC) 1
-
/* Specify the machine mode that pointers have.
After generation of rtl, the compiler makes no further distinction
between pointers and any other objects of this machine mode. */
of a shift count. */
/* #define SHIFT_COUNT_TRUNCATED */
-/* Value is 1 if truncating an integer of INPREC bits to OUTPREC bits
- is done just by pretending it is already truncated. */
-#define TRULY_NOOP_TRUNCATION(OUTPREC, INPREC) 1
-
/* Specify the machine mode that pointers have.
After generation of rtl, the compiler makes no further distinction
between pointers and any other objects of this machine mode. */
bitfield instructions. */
#define SHIFT_COUNT_TRUNCATED 0
-/* `TRULY_NOOP_TRUNCATION (OUTPREC, INPREC)'
-
- A C expression which is nonzero if on this machine it is safe to
- "convert" an integer of INPREC bits to one of OUTPREC bits (where
- OUTPREC is smaller than INPREC) by merely operating on it as if it
- had only OUTPREC bits.
-
- On many machines, this expression can be 1.
-
- When `TRULY_NOOP_TRUNCATION' returns 1 for a pair of sizes for
- modes for which `TARGET_MODES_TIEABLE_P' is 0, suboptimal code can result.
- If this is the case, making `TRULY_NOOP_TRUNCATION' return 0 in
- such cases may improve things. */
-#define TRULY_NOOP_TRUNCATION(OUTPREC, INPREC) 1
-
/* `STORE_FLAG_VALUE'
A C expression describing the value returned by a comparison
/* Shift instructions ignore all but the low-order few bits. */
#define SHIFT_COUNT_TRUNCATED 1
-/* Value is 1 if truncating an integer of INPREC bits to OUTPREC bits
- is done just by pretending it is already truncated. */
-#define TRULY_NOOP_TRUNCATION(OUTPREC, INPREC) 1
-
#define CLZ_DEFINED_VALUE_AT_ZERO(MODE, VALUE) ((VALUE) = 32, 1)
#define CTZ_DEFINED_VALUE_AT_ZERO(MODE, VALUE) ((VALUE) = -1, 1)
break;
if (outprec >= BITS_PER_WORD
- || TRULY_NOOP_TRUNCATION (outprec, inprec)
+ || targetm.truly_noop_truncation (outprec, inprec)
|| inprec > TYPE_PRECISION (TREE_TYPE (arg0))
|| inprec > TYPE_PRECISION (TREE_TYPE (arg1)))
{
by overriding it.
@end deftypefn
-@defmac TRULY_NOOP_TRUNCATION (@var{outprec}, @var{inprec})
-A C expression which is nonzero if on this machine it is safe to
-``convert'' an integer of @var{inprec} bits to one of @var{outprec}
-bits (where @var{outprec} is smaller than @var{inprec}) by merely
-operating on it as if it had only @var{outprec} bits.
+@deftypefn {Target Hook} bool TARGET_TRULY_NOOP_TRUNCATION (unsigned int @var{outprec}, unsigned int @var{inprec})
+This hook returns true if it is safe to ``convert'' a value of
+@var{inprec} bits to one of @var{outprec} bits (where @var{outprec} is
+smaller than @var{inprec}) by merely operating on it as if it had only
+@var{outprec} bits. The default returns true unconditionally, which
+is correct for most machines.
-On many machines, this expression can be 1.
-
-@c rearranged this, removed the phrase "it is reported that". this was
-@c to fix an overfull hbox. --mew 10feb93
-When @code{TRULY_NOOP_TRUNCATION} returns 1 for a pair of sizes for modes
-for which @code{TARGET_MODES_TIEABLE_P} is false, suboptimal code can result.
-If this is the case, making @code{TRULY_NOOP_TRUNCATION} return 0 in
-such cases may improve things.
-@end defmac
+If @code{TARGET_MODES_TIEABLE_P} returns false for a pair of modes,
+suboptimal code can result if this hook returns true for the corresponding
+mode sizes. Making this hook return false in such cases may improve things.
+@end deftypefn
@deftypefn {Target Hook} int TARGET_MODE_REP_EXTENDED (scalar_int_mode @var{mode}, scalar_int_mode @var{rep_mode})
The representation of an integral mode can be such that the values
extension.
In order to enforce the representation of @code{mode},
-@code{TRULY_NOOP_TRUNCATION} should return false when truncating to
+@code{TARGET_TRULY_NOOP_TRUNCATION} should return false when truncating to
@code{mode}.
@end deftypefn
@anchor{TARGET_SHIFT_TRUNCATION_MASK}
@hook TARGET_SHIFT_TRUNCATION_MASK
-@defmac TRULY_NOOP_TRUNCATION (@var{outprec}, @var{inprec})
-A C expression which is nonzero if on this machine it is safe to
-``convert'' an integer of @var{inprec} bits to one of @var{outprec}
-bits (where @var{outprec} is smaller than @var{inprec}) by merely
-operating on it as if it had only @var{outprec} bits.
-
-On many machines, this expression can be 1.
-
-@c rearranged this, removed the phrase "it is reported that". this was
-@c to fix an overfull hbox. --mew 10feb93
-When @code{TRULY_NOOP_TRUNCATION} returns 1 for a pair of sizes for modes
-for which @code{TARGET_MODES_TIEABLE_P} is false, suboptimal code can result.
-If this is the case, making @code{TRULY_NOOP_TRUNCATION} return 0 in
-such cases may improve things.
-@end defmac
+@hook TARGET_TRULY_NOOP_TRUNCATION
@hook TARGET_MODE_REP_EXTENDED
&& !reverse
/* ??? We could limit the structure size to the part of OP0 that
contains the field, with appropriate checks for endianness
- and TRULY_NOOP_TRUNCATION. */
+ and TARGET_TRULY_NOOP_TRUNCATION. */
&& get_best_reg_extraction_insn (&extv, pattern,
GET_MODE_BITSIZE (op0_mode.require ()),
tmode))
a zero extension
- when MODE is smaller than SRC_MODE, the extraction involves
- a truncation (and is thus subject to TRULY_NOOP_TRUNCATION).
+ a truncation (and is thus subject to TARGET_TRULY_NOOP_TRUNCATION).
In other words, this routine performs a computation, whereas the
gen_lowpart* routines are conceptually lvalue or rvalue subreg
to the value directly in mode MODE, otherwise we must
start with the register in word_mode and explicitly
convert it. */
- if (TRULY_NOOP_TRUNCATION (size * BITS_PER_UNIT, BITS_PER_WORD))
+ if (targetm.truly_noop_truncation (size * BITS_PER_UNIT,
+ BITS_PER_WORD))
reg = gen_rtx_REG (mode, REGNO (entry_parm));
else
{
return false;
}
+/* Generic hook that takes (unsigned int, unsigned int) and returns true. */
+bool
+hook_bool_uint_uint_true (unsigned int, unsigned int)
+{
+ return true;
+}
+
/* Generic hook that takes (unsigned int, machine_mode) and returns false. */
bool
hook_bool_uint_mode_false (unsigned int, machine_mode)
const rtx_insn *);
extern bool hook_bool_mode_uhwi_false (machine_mode,
unsigned HOST_WIDE_INT);
+extern bool hook_bool_uint_uint_true (unsigned int, unsigned int);
extern bool hook_bool_uint_mode_false (unsigned int, machine_mode);
extern bool hook_bool_uint_mode_true (unsigned int, machine_mode);
extern bool hook_bool_tree_false (tree);
extern void init_adjust_machine_modes (void);
#define TRULY_NOOP_TRUNCATION_MODES_P(MODE1, MODE2) \
- TRULY_NOOP_TRUNCATION (GET_MODE_PRECISION (MODE1), \
- GET_MODE_PRECISION (MODE2))
+ (targetm.truly_noop_truncation (GET_MODE_PRECISION (MODE1), \
+ GET_MODE_PRECISION (MODE2)))
#define HWI_COMPUTABLE_MODE_P(MODE) \
(SCALAR_INT_MODE_P (MODE) \
#include "recog.h"
#include "rtlhooks-def.h"
#include "explow.h"
+#include "target.h"
\f
/* For speed, we will copy the RTX hooks struct member-by-member
HARD_REGNO_CALL_PART_CLOBBERED HARD_REGNO_MODE_OK \
MODES_TIEABLE_P FUNCTION_ARG_PADDING SLOW_UNALIGNED_ACCESS \
HARD_REGNO_NREGS SECONDARY_MEMORY_NEEDED_MODE \
- SECONDARY_MEMORY_NEEDED CANNOT_CHANGE_MODE_CLASS
+ SECONDARY_MEMORY_NEEDED CANNOT_CHANGE_MODE_CLASS \
+ TRULY_NOOP_TRUNCATION
/* Target macros only used for code built for the target, that have
moved to libgcc-tm.h or have never been present elsewhere. */
unsigned int, (machine_mode mode),
default_min_divisions_for_recip_mul)
+DEFHOOK
+(truly_noop_truncation,
+ "This hook returns true if it is safe to ``convert'' a value of\n\
+@var{inprec} bits to one of @var{outprec} bits (where @var{outprec} is\n\
+smaller than @var{inprec}) by merely operating on it as if it had only\n\
+@var{outprec} bits. The default returns true unconditionally, which\n\
+is correct for most machines.\n\
+\n\
+If @code{TARGET_MODES_TIEABLE_P} returns false for a pair of modes,\n\
+suboptimal code can result if this hook returns true for the corresponding\n\
+mode sizes. Making this hook return false in such cases may improve things.",
+ bool, (unsigned int outprec, unsigned int inprec),
+ hook_bool_uint_uint_true)
+
/* If the representation of integral MODE is such that values are
always sign-extended to a wider mode MODE_REP then return
SIGN_EXTEND. Return UNKNOWN otherwise. */
extension.\n\
\n\
In order to enforce the representation of @code{mode},\n\
-@code{TRULY_NOOP_TRUNCATION} should return false when truncating to\n\
+@code{TARGET_TRULY_NOOP_TRUNCATION} should return false when truncating to\n\
@code{mode}.",
int, (scalar_int_mode mode, scalar_int_mode rep_mode),
default_mode_rep_extended)