PR 70687: Use wide_int in combine.c:change_zero_ext
authorRichard Sandiford <richard.sandiford@arm.com>
Tue, 3 May 2016 10:29:55 +0000 (10:29 +0000)
committerRichard Sandiford <rsandifo@gcc.gnu.org>
Tue, 3 May 2016 10:29:55 +0000 (10:29 +0000)
PR 70687 reports a case where combine.c mishandles integer modes
wider than unsigned HOST_WIDE_INT.  I don't have a testcase since
the PR is just pointing out the hole.

Also, I think a ZERO_EXTEND of a vector mode could in principle satisfy
the subreg condition but wouldn't be equivalent to an AND.  E.g.:

      (zero_extend:V4DI (subreg:V4SI (reg:V4DI R) 0))

Tested on x86_64-linux-gnu.

gcc/
PR rtl-optimization/70687
* combine.c (change_zero_ext): Check for scalar modes.  Use wide_int
instead of unsigned HOST_WIDE_INT.

From-SVN: r235811

gcc/ChangeLog
gcc/combine.c

index 608f64b63898d870646fd164bd866ef8a46516b9..cc520e895efa7f0c9a43e97a2bfa5da311a63e53 100644 (file)
@@ -1,3 +1,9 @@
+2016-05-03  Richard Sandiford  <richard.sandiford@arm.com>
+
+       PR rtl-optimization/70687
+       * combine.c (change_zero_ext): Check for scalar modes.  Use wide_int
+       instead of unsigned HOST_WIDE_INT.
+
 2016-05-03  Bernd Schmidt  <bschmidt@redhat.com>
 
        PR rtl-optimization/44281
index 1d0e8beeb2c07b50443b6a22103802cba075b088..0ab3f97db697b723dbfcfda0537269c0367cf0b3 100644 (file)
@@ -11096,6 +11096,7 @@ change_zero_ext (rtx *src)
                                   XEXP (x, 0), GEN_INT (start));
        }
       else if (GET_CODE (x) == ZERO_EXTEND
+              && SCALAR_INT_MODE_P (mode)
               && GET_CODE (XEXP (x, 0)) == SUBREG
               && GET_MODE (SUBREG_REG (XEXP (x, 0))) == mode
               && subreg_lowpart_p (XEXP (x, 0)))
@@ -11106,11 +11107,8 @@ change_zero_ext (rtx *src)
       else
        continue;
 
-      unsigned HOST_WIDE_INT mask = 1;
-      mask <<= size;
-      mask--;
-
-      x = gen_rtx_AND (mode, x, GEN_INT (mask));
+      wide_int mask = wi::mask (size, false, GET_MODE_PRECISION (mode));
+      x = gen_rtx_AND (mode, x, immed_wide_int_const (mask, mode));
 
       SUBST (**iter, x);
       changed = true;