Fix shift count truncate problem for loongson.

From-SVN: r163799
This commit is contained in:
Mingjie Xing 2010-09-03 09:29:19 +00:00 committed by Mingjie Xing
parent e5ca969363
commit 4904231323
5 changed files with 66 additions and 3 deletions

View File

@ -1,3 +1,9 @@
2010-09-03 Mingjie Xing <mingjie.xing@gmail.com>
* config/mips/mips.h (SHIFT_COUNT_TRUNCATED): Change the definition.
* config/mips/mips.c (mips_shift_truncation_mask): New function.
(TARGET_SHIFT_TRUNCATION_MASK): Define.
2010-09-02 Richard Henderson <rth@redhat.com> 2010-09-02 Richard Henderson <rth@redhat.com>
* configure.ac (gcc_cv_as_cfi_advance_working): Use objdump * configure.ac (gcc_cv_as_cfi_advance_working): Use objdump

View File

@ -16338,6 +16338,20 @@ void mips_function_profiler (FILE *file)
fprintf (file, "\tmove\t%s,%s\n", reg_names[STATIC_CHAIN_REGNUM], fprintf (file, "\tmove\t%s,%s\n", reg_names[STATIC_CHAIN_REGNUM],
reg_names[2]); reg_names[2]);
} }
/* Implement TARGET_SHIFT_TRUNCATION_MASK. We want to keep the default
behaviour of TARGET_SHIFT_TRUNCATION_MASK for non-vector modes even
when TARGET_LOONGSON_2EF is true. */
static unsigned HOST_WIDE_INT
mips_shift_truncation_mask (enum machine_mode mode)
{
if (TARGET_LOONGSON_2EF && VECTOR_MODE_P (mode))
return 0;
return GET_MODE_BITSIZE (mode) - 1;
}
/* Initialize the GCC target structure. */ /* Initialize the GCC target structure. */
#undef TARGET_ASM_ALIGNED_HI_OP #undef TARGET_ASM_ALIGNED_HI_OP
@ -16546,6 +16560,9 @@ void mips_function_profiler (FILE *file)
#undef TARGET_ASM_OUTPUT_SOURCE_FILENAME #undef TARGET_ASM_OUTPUT_SOURCE_FILENAME
#define TARGET_ASM_OUTPUT_SOURCE_FILENAME mips_output_filename #define TARGET_ASM_OUTPUT_SOURCE_FILENAME mips_output_filename
#undef TARGET_SHIFT_TRUNCATION_MASK
#define TARGET_SHIFT_TRUNCATION_MASK mips_shift_truncation_mask
struct gcc_target targetm = TARGET_INITIALIZER; struct gcc_target targetm = TARGET_INITIALIZER;
#include "gt-mips.h" #include "gt-mips.h"

View File

@ -2423,9 +2423,10 @@ typedef struct mips_args {
(often extended) would be needed for byte accesses. */ (often extended) would be needed for byte accesses. */
#define SLOW_BYTE_ACCESS (!TARGET_MIPS16) #define SLOW_BYTE_ACCESS (!TARGET_MIPS16)
/* Define this to be nonzero if shift instructions ignore all but the low-order /* Standard MIPS integer shifts truncate the shift amount to the
few bits. */ width of the shifted operand. However, Loongson vector shifts
#define SHIFT_COUNT_TRUNCATED 1 do not truncate the shift amount at all. */
#define SHIFT_COUNT_TRUNCATED (!TARGET_LOONGSON_2EF)
/* Value is 1 if truncating an integer of INPREC bits to OUTPREC bits /* Value is 1 if truncating an integer of INPREC bits to OUTPREC bits
is done just by pretending it is already truncated. */ is done just by pretending it is already truncated. */

View File

@ -1,3 +1,7 @@
2010-09-03 Mingjie Xing <mingjie.xing@gmail.com>
* gcc.target/mips/loongson-shift-count-truncated-1.c: New.
2010-09-03 Daniel Kraft <d@domob.eu> 2010-09-03 Daniel Kraft <d@domob.eu>
PR fortran/44602 PR fortran/44602

View File

@ -0,0 +1,35 @@
/* Test case for SHIFT_COUNT_TRUNCATED on Loongson. */
/* { dg-do run } */
/* loongson.h does not handle or check for MIPS16ness. There doesn't
seem any good reason for it to, given that the Loongson processors
do not support MIPS16. */
/* { dg-options "isa=loongson -mhard-float -mno-mips16 -O1" } */
#include "loongson.h"
#include <assert.h>
typedef union { int32x2_t v; int32_t a[2]; } int32x2_encap_t;
void
main1 (int shift)
{
int32x2_encap_t s;
int32x2_encap_t r;
s.a[0] = 0xffffffff;
s.a[1] = 0xffffffff;
/* Loongson SIMD use low-order 7 bits to specify the shift amount.
Thus V2SI << 0x40 == 0. The below expression 'shift & 0x3f' will be
mis-optimized as 'shift', if SHIFT_COUNT_TRUNCATED is nonzero. */
r.v = psllw_s (s.v, (shift & 0x3f));
assert (r.a[0] == 0xffffffff);
assert (r.a[1] == 0xffffffff);
}
int
main (void)
{
main1 (0x40);
return 0;
}