match.pd: Simplify X ? C : 0 where C is a power of 2 and X tests a single bit.
authorPaolo Bonzini <bonzini@gcc.gnu.org>
Mon, 5 Dec 2016 13:19:34 +0000 (13:19 +0000)
committerPaolo Bonzini <bonzini@gcc.gnu.org>
Mon, 5 Dec 2016 13:19:34 +0000 (13:19 +0000)
gcc:
* match.pd: Simplify X ? C : 0 where C is a power of 2 and
X tests a single bit.

gcc/testsuite:
* gcc.dg/fold-and-lshift.c, gcc.dg/fold-and-rshift-1.c,
gcc.dg/fold-and-rshift-2.c: New testcases.

From-SVN: r243255

gcc/ChangeLog
gcc/match.pd
gcc/testsuite/ChangeLog
gcc/testsuite/gcc.dg/fold-and-lshift.c [new file with mode: 0644]
gcc/testsuite/gcc.dg/fold-and-rshift-1.c [new file with mode: 0644]
gcc/testsuite/gcc.dg/fold-and-rshift-2.c [new file with mode: 0644]

index 9488b0f7d9e1eb8facfc2ae7d908943fba5f0b63..2ba03023cb075d011684eba4dec446baa4ab456a 100644 (file)
@@ -1,3 +1,8 @@
+2016-12-05  Paolo Bonzini  <bonzini@gnu.org>
+
+       * match.pd: Simplify X ? C : 0 where C is a power of 2 and
+       X tests a single bit.
+
 2016-12-05  Nathan Sidwell  <nathan@acm.org>
 
        * diagnostic.c (diagnostic_check_max_errors): New, broken out of ...
@@ -25,7 +30,7 @@
        * config/arc/arc.md (ls_gd_load): Remove unused pattern.
        (tls_gd_dispatch): Likewise.
 
-2016-12-025  Andre Vieira  <andre.simoesdiasvieira@arm.com>
+2016-12-05  Andre Vieira  <andre.simoesdiasvieira@arm.com>
 
        * config/arm/arm.c (TARGET_ASM_INIT_SECTIONS): Fix wrong undef
        location.
index dbb91034e95a3b71af5b717cec86a0025a0c8cf6..1fe003ba1582cf26df710915d164aa7b5dbd71c2 100644 (file)
@@ -2737,6 +2737,21 @@ DEFINE_INT_AND_FLOAT_ROUND_FN (RINT)
   (cmp (bit_and@2 @0 integer_pow2p@1) @1)
   (icmp @2 { build_zero_cst (TREE_TYPE (@0)); })))
  
+/* If we have (A & C) != 0 ? D : 0 where C and D are powers of 2,
+   convert this into a shift followed by ANDing with D.  */
+(simplify
+ (cond
+  (ne (bit_and @0 integer_pow2p@1) integer_zerop)
+  integer_pow2p@2 integer_zerop)
+ (with {
+    int shift = wi::exact_log2 (@2) - wi::exact_log2 (@1);
+  }
+  (if (shift > 0)
+   (bit_and
+    (lshift (convert @0) { build_int_cst (integer_type_node, shift); }) @2)
+   (bit_and
+    (convert (rshift @0 { build_int_cst (integer_type_node, -shift); })) @2))))
+
 /* If we have (A & C) != 0 where C is the sign bit of A, convert
    this into A < 0.  Similarly for (A & C) == 0 into A >= 0.  */
 (for cmp (eq ne)
@@ -2751,6 +2766,19 @@ DEFINE_INT_AND_FLOAT_ROUND_FN (RINT)
    (with { tree stype = signed_type_for (TREE_TYPE (@0)); }
     (ncmp (convert:stype @0) { build_zero_cst (stype); })))))
 
+/* If we have A < 0 ? C : 0 where C is a power of 2, convert
+   this into a right shift followed by ANDing with C.  */
+(simplify
+ (cond
+  (lt @0 integer_zerop)
+  integer_pow2p@1 integer_zerop)
+ (with {
+    int shift = element_precision (@0) - wi::exact_log2 (@1) - 1;
+  }
+  (bit_and
+   (convert (rshift @0 { build_int_cst (integer_type_node, shift); }))
+   @1)))
+
 /* When the addresses are not directly of decls compare base and offset.
    This implements some remaining parts of fold_comparison address
    comparisons but still no complete part of it.  Still it is good
index c40ffd60b21191874770221ca51929b4e4e5a776..d9edb52d3b2430aa78a1f08834c6de2d57305227 100644 (file)
@@ -1,3 +1,8 @@
+2016-12-05  Paolo Bonzini  <bonzini@gnu.org>
+
+       * gcc.dg/fold-and-lshift.c, gcc.dg/fold-and-rshift-1.c,
+       gcc.dg/fold-and-rshift-2.c: New testcases.
+
 2016-12-05  Nathan Sidwell  <nathan@acm.org>
 
        * c-c++-common/fmax_errors.c: Check notes after last error are
diff --git a/gcc/testsuite/gcc.dg/fold-and-lshift.c b/gcc/testsuite/gcc.dg/fold-and-lshift.c
new file mode 100644 (file)
index 0000000..2905095
--- /dev/null
@@ -0,0 +1,35 @@
+/* { dg-do compile } */\r
+/* { dg-options "-O -fdump-tree-original" } */\r
+\r
+int f(int x)\r
+{\r
+       return (x << 2) & 128;\r
+}\r
+\r
+int g(int x)\r
+{\r
+       return !!(x & 32) << 7;\r
+}\r
+\r
+int h(int x)\r
+{\r
+       return ((x >> 5) & 1) << 7;\r
+}\r
+\r
+int i(int x)\r
+{\r
+       return (x & 32) >> 5 << 7;\r
+}\r
+\r
+int j(int x)\r
+{\r
+       return ((x >> 5) & 1) ? 128 : 0;\r
+}\r
+\r
+int k(int x)\r
+{\r
+       return (x & 32) ? 128 : 0;\r
+}\r
+\r
+/* { dg-final { scan-tree-dump-not " \\? " "original" } } */\r
+/* { dg-final { scan-assembler-not "sarl" { target i?86-*-* x86_64-*-* } } }" */\r
diff --git a/gcc/testsuite/gcc.dg/fold-and-rshift-1.c b/gcc/testsuite/gcc.dg/fold-and-rshift-1.c
new file mode 100644 (file)
index 0000000..11c13d5
--- /dev/null
@@ -0,0 +1,35 @@
+/* { dg-do compile } */\r
+/* { dg-options "-O -fdump-tree-original" } */\r
+\r
+int f(int x)\r
+{\r
+       return (x >> 2) & 128;\r
+}\r
+\r
+int g(int x)\r
+{\r
+       return !!(x & 512) << 7;\r
+}\r
+\r
+int h(int x)\r
+{\r
+       return ((x >> 9) & 1) << 7;\r
+}\r
+\r
+int i(int x)\r
+{\r
+       return (x & 512) >> 9 << 7;\r
+}\r
+\r
+int j(int x)\r
+{\r
+       return ((x >> 9) & 1) ? 128 : 0;\r
+}\r
+\r
+int k(int x)\r
+{\r
+       return (x & 512) ? 128 : 0;\r
+}\r
+\r
+/* { dg-final { scan-tree-dump-not " \\? " "original" } } */\r
+/* { dg-final { scan-assembler-not "sall" { target i?86-*-* x86_64-*-* } } }" */\r
diff --git a/gcc/testsuite/gcc.dg/fold-and-rshift-2.c b/gcc/testsuite/gcc.dg/fold-and-rshift-2.c
new file mode 100644 (file)
index 0000000..f88d48d
--- /dev/null
@@ -0,0 +1,25 @@
+/* { dg-do compile } */\r
+/* { dg-options "-O -fdump-tree-original" } */\r
+\r
+unsigned f(unsigned x)\r
+{\r
+       return (x >> 29) & 32;\r
+}\r
+\r
+unsigned g(unsigned x)\r
+{\r
+       return !!(x & 0x80000000) << 5;\r
+}\r
+\r
+unsigned j(unsigned x)\r
+{\r
+       return ((x >> 31) & 1) ? 32 : 0;\r
+}\r
+\r
+unsigned k(unsigned x)\r
+{\r
+       return (x & 0x80000000) ? 32 : 0;\r
+}\r
+\r
+/* { dg-final { scan-tree-dump-not " \\? " "original" } } */\r
+/* { dg-final { scan-assembler-not "sall" { target i?86-*-* x86_64-*-* } } }" */\r