diff --git a/gcc/match.pd b/gcc/match.pd index 6991868fbe29fec1759f0d9a65b938a805489b00..97e0bafdda4b9aefc4d8be07d9c78b3cbebc4204 100644 --- a/gcc/match.pd +++ b/gcc/match.pd @@ -303,6 +303,23 @@ DEFINE_INT_AND_FLOAT_ROUND_FN (RINT) (mult @0 integer_zerop@1) @1) +#if GIMPLE +/* When multiplying a value by a boolean involving the value, we may + be able to simplify further. + a * ((a || b) != 0) -> a + a * ((a || b) == 0) -> 0 + + There are also bit-and cases which don't show up in practice yet. + a * ((a && b) != 0) -> a * b + a * ((a && b) == 0) -> b != 0 ? a : b */ +(for neeq (ne eq) + (simplify + (mult:c (convert? (neeq (bit_ior:c @0 @1) integer_zerop@2)) @0) + (if (neeq == EQ_EXPR) + { build_zero_cst (type); } + @0))) +#endif + /* -x == x -> x == 0 */ (for cmp (eq ne) (simplify diff --git a/gcc/testsuite/gcc.target/i386/pr114277.c b/gcc/testsuite/gcc.target/i386/pr114277.c new file mode 100644 index 0000000000000000000000000000000000000000..eb611d26d6af1b8b27fb1dae00995656970a3a08 --- /dev/null +++ b/gcc/testsuite/gcc.target/i386/pr114277.c @@ -0,0 +1,10 @@ +/* { dg-do compile } */ +/* { dg-options "-O2" } */ + +int a,b; +void func0(int x) { a=x * (x || b); } +void func1(int x) { a=x * !(x || b); } + +/* { dg-final { scan-assembler-not "or" } } */ +/* { dg-final { scan-assembler-not "cmove" } } */ + diff --git a/gcc/testsuite/gcc.target/riscv/pr114277.c b/gcc/testsuite/gcc.target/riscv/pr114277.c new file mode 100644 index 0000000000000000000000000000000000000000..cc1db19ee4c5518730ef947d468e321b4ddcac1e --- /dev/null +++ b/gcc/testsuite/gcc.target/riscv/pr114277.c @@ -0,0 +1,9 @@ +/* { dg-do compile } */ +/* { dg-options "-O2 -march=rv64gc_zicond -mabi=lp64d" { target rv64 } } */ +/* { dg-options "-O2 -march=rv32gc_zicond -mabi=ilp32" { target rv32 } } */ +/* { dg-skip-if "" { *-*-* } { "-O0" "-O1" "-Os" "-Oz" "-O3" "-Og" } } */ + +#include "../i386/pr114277.c" + +/* { dg-final { scan-assembler-not "czero" } } */ +