diff options
Diffstat (limited to 'gcc')
| -rw-r--r-- | gcc/ChangeLog | 7 | ||||
| -rw-r--r-- | gcc/reload1.c | 17 | ||||
| -rw-r--r-- | gcc/testsuite/ChangeLog | 4 | ||||
| -rw-r--r-- | gcc/testsuite/gcc.dg/torture/pr19683-1.c | 42 | 
4 files changed, 61 insertions, 9 deletions
| diff --git a/gcc/ChangeLog b/gcc/ChangeLog index 9c6b4483e04..86bc853c1c0 100644 --- a/gcc/ChangeLog +++ b/gcc/ChangeLog @@ -1,3 +1,10 @@ +2005-03-07  Richard Sandiford  <rsandifo@redhat.com> + +	PR rtl-optimization/19683 +	* reload1.c (choose_reload_regs): Pass the number of bits, not the +	number of bytes, to smallest_int_for_mode.  Fix arguments to +	REG_CANNOT_CHANGE_MODE_P. +  2005-03-07  Eric Botcazou  <ebotcazou@libertysurf.fr>  	* reorg.c (relax_delay_slots): Check that the jump is diff --git a/gcc/reload1.c b/gcc/reload1.c index 32f8ddcd22e..31407b3d96f 100644 --- a/gcc/reload1.c +++ b/gcc/reload1.c @@ -5411,19 +5411,18 @@ choose_reload_regs (struct insn_chain *chain)  		    need_mode = mode;  		  else  		    need_mode -		      = smallest_mode_for_size (GET_MODE_SIZE (mode) + byte, +		      = smallest_mode_for_size (GET_MODE_BITSIZE (mode) +						+ byte * BITS_PER_UNIT,  						GET_MODE_CLASS (mode)); -		  if ( -#ifdef CANNOT_CHANGE_MODE_CLASS -		      (!REG_CANNOT_CHANGE_MODE_P (i, GET_MODE (last_reg), -						  need_mode) -		       && -#endif -		      (GET_MODE_SIZE (GET_MODE (last_reg)) +		  if ((GET_MODE_SIZE (GET_MODE (last_reg))  		       >= GET_MODE_SIZE (need_mode))  #ifdef CANNOT_CHANGE_MODE_CLASS -		      ) +		      /* Verify that the register in "i" can be obtained +			 from LAST_REG.  */ +		      && !REG_CANNOT_CHANGE_MODE_P (REGNO (last_reg), +						    GET_MODE (last_reg), +						    mode)  #endif  		      && reg_reloaded_contents[i] == regno  		      && TEST_HARD_REG_BIT (reg_reloaded_valid, i) diff --git a/gcc/testsuite/ChangeLog b/gcc/testsuite/ChangeLog index 1abdca14526..3b330421f4b 100644 --- a/gcc/testsuite/ChangeLog +++ b/gcc/testsuite/ChangeLog @@ -1,3 +1,7 @@ +2005-03-07  Richard Sandiford  <rsandifo@redhat.com> + +	* gcc.dg/torture/pr19683-1.c: New test. +  2005-03-06  Steven G. Kargl  <kargls@comcast.net>  	* gfortran.dg/g77/19990313-1.f: Replace tabs with spaces. diff --git a/gcc/testsuite/gcc.dg/torture/pr19683-1.c b/gcc/testsuite/gcc.dg/torture/pr19683-1.c new file mode 100644 index 00000000000..4015fb981e9 --- /dev/null +++ b/gcc/testsuite/gcc.dg/torture/pr19683-1.c @@ -0,0 +1,42 @@ +/* From PR rtl-optimization/19683.  On little-endian MIPS targets, +   reload would incorrectly inherit the high part of the multiplication +   result.  */ +/* { dg-do run { target mips*-*-* } } */ + +extern void abort (void); +extern void exit (int); + +#define REPEAT10(X, Y)					\ +  X(Y##0); X(Y##1); X(Y##2); X(Y##3); X(Y##4);		\ +  X(Y##5); X(Y##6); X(Y##7); X(Y##8); X(Y##9) + +#define REPEAT30(X) REPEAT10 (X, 0); REPEAT10 (X, 1); REPEAT10 (X, 2) +#define IN(X) unsigned int x##X = ptr[0] +#define OUT(X) ptr[0] = x##X + +union u { unsigned long long ll; unsigned int i[2]; }; + +unsigned int +foo (volatile unsigned int *ptr) +{ +  union u u; +  int result; + +  u.ll = (unsigned long long) ptr[0] * ptr[0]; +  REPEAT30 (IN); +  REPEAT30 (OUT); +  asm ("#" : "=l" (result) : "l" (u.i[1])); +  return result; +} + +int +main (void) +{ +  unsigned int array[] = { 1000 * 1000 * 1000 }; +  union u u; + +  u.ll = (unsigned long long) array[0] * array[0]; +  if (foo (array) != u.i[1]) +    abort (); +  exit (0); +} | 

