* [PATCH 1/2] MATCH: [PR111364] Add some more minmax cmp operand simplifications
@ 2023-09-12 15:25 Andrew Pinski
2023-09-12 15:25 ` [PATCH 2/2] MATCH: Move `X <= MAX(X, Y)` before `MIN (X, C1) < C2` pattern Andrew Pinski
2023-09-13 6:42 ` [PATCH 1/2] MATCH: [PR111364] Add some more minmax cmp operand simplifications Richard Biener
0 siblings, 2 replies; 5+ messages in thread
From: Andrew Pinski @ 2023-09-12 15:25 UTC (permalink / raw)
To: gcc-patches; +Cc: Andrew Pinski
This adds a few more minmax cmp operand simplifications which were missed before.
`MIN(a,b) < a` -> `a > b`
`MIN(a,b) >= a` -> `a <= b`
`MAX(a,b) > a` -> `a < b`
`MAX(a,b) <= a` -> `a >= b`
OK? Bootstrapped and tested on x86_64-linux-gnu.
Note gcc.dg/pr96708-negative.c needed to updated to remove the
check for MIN/MAX as they have been optimized (correctly) away.
PR tree-optimization/111364
gcc/ChangeLog:
* match.pd (`MIN (X, Y) == X`): Extend
to min/lt, min/ge, max/gt, max/le.
gcc/testsuite/ChangeLog:
* gcc.c-torture/execute/minmaxcmp-1.c: New test.
* gcc.dg/tree-ssa/minmaxcmp-2.c: New test.
* gcc.dg/pr96708-negative.c: Update testcase.
* gcc.dg/pr96708-positive.c: Add comment about `return 0`.
---
gcc/match.pd | 8 +--
.../gcc.c-torture/execute/minmaxcmp-1.c | 51 +++++++++++++++++++
gcc/testsuite/gcc.dg/pr96708-negative.c | 4 +-
gcc/testsuite/gcc.dg/pr96708-positive.c | 1 +
gcc/testsuite/gcc.dg/tree-ssa/minmaxcmp-2.c | 30 +++++++++++
5 files changed, 89 insertions(+), 5 deletions(-)
create mode 100644 gcc/testsuite/gcc.c-torture/execute/minmaxcmp-1.c
create mode 100644 gcc/testsuite/gcc.dg/tree-ssa/minmaxcmp-2.c
diff --git a/gcc/match.pd b/gcc/match.pd
index 51985c1bad4..36e3da4841b 100644
--- a/gcc/match.pd
+++ b/gcc/match.pd
@@ -3902,9 +3902,11 @@ DEFINE_INT_AND_FLOAT_ROUND_FN (RINT)
(maxmin @0 (bit_not @1))))
/* MIN (X, Y) == X -> X <= Y */
-(for minmax (min min max max)
- cmp (eq ne eq ne )
- out (le gt ge lt )
+/* MIN (X, Y) < X -> X > Y */
+/* MIN (X, Y) >= X -> X <= Y */
+(for minmax (min min min min max max max max)
+ cmp (eq ne lt ge eq ne gt le )
+ out (le gt gt le ge lt lt ge )
(simplify
(cmp:c (minmax:c @0 @1) @0)
(if (ANY_INTEGRAL_TYPE_P (TREE_TYPE (@0)))
diff --git a/gcc/testsuite/gcc.c-torture/execute/minmaxcmp-1.c b/gcc/testsuite/gcc.c-torture/execute/minmaxcmp-1.c
new file mode 100644
index 00000000000..6705a053768
--- /dev/null
+++ b/gcc/testsuite/gcc.c-torture/execute/minmaxcmp-1.c
@@ -0,0 +1,51 @@
+#define func(vol, op1, op2) \
+_Bool op1##_##op2##_##vol (int a, int b) \
+{ \
+ vol int x = op_##op1(a, b); \
+ return op_##op2(x, a); \
+}
+
+#define op_lt(a, b) ((a) < (b))
+#define op_le(a, b) ((a) <= (b))
+#define op_eq(a, b) ((a) == (b))
+#define op_ne(a, b) ((a) != (b))
+#define op_gt(a, b) ((a) > (b))
+#define op_ge(a, b) ((a) >= (b))
+#define op_min(a, b) ((a) < (b) ? (a) : (b))
+#define op_max(a, b) ((a) > (b) ? (a) : (b))
+
+
+#define funcs(a) \
+ a(min,lt) \
+ a(max,lt) \
+ a(min,gt) \
+ a(max,gt) \
+ a(min,le) \
+ a(max,le) \
+ a(min,ge) \
+ a(max,ge) \
+ a(min,ne) \
+ a(max,ne) \
+ a(min,eq) \
+ a(max,eq)
+
+#define funcs1(a,b) \
+func(,a,b) \
+func(volatile,a,b)
+
+funcs(funcs1)
+
+#define test(op1,op2) \
+do { \
+ if (op1##_##op2##_(x,y) != op1##_##op2##_volatile(x,y)) \
+ __builtin_abort(); \
+} while(0);
+
+int main()
+{
+ for(int x = -10; x < 10; x++)
+ for(int y = -10; y < 10; y++)
+ {
+ funcs(test)
+ }
+}
diff --git a/gcc/testsuite/gcc.dg/pr96708-negative.c b/gcc/testsuite/gcc.dg/pr96708-negative.c
index 91964d3b971..c9c1aa85558 100644
--- a/gcc/testsuite/gcc.dg/pr96708-negative.c
+++ b/gcc/testsuite/gcc.dg/pr96708-negative.c
@@ -42,7 +42,7 @@ int main()
return 0;
}
-/* { dg-final { scan-tree-dump-times "MAX_EXPR" 2 "optimized" } } */
-/* { dg-final { scan-tree-dump-times "MIN_EXPR" 2 "optimized" } } */
+/* Even though test[1-4] originally has MIN/MAX, those can be optimized away
+ into just comparing a and b arguments. */
/* { dg-final { scan-tree-dump-times "return 0;" 1 "optimized" } } */
/* { dg-final { scan-tree-dump-not { "return 1;" } "optimized" } } */
diff --git a/gcc/testsuite/gcc.dg/pr96708-positive.c b/gcc/testsuite/gcc.dg/pr96708-positive.c
index 65af85344b6..12c5fedfd30 100644
--- a/gcc/testsuite/gcc.dg/pr96708-positive.c
+++ b/gcc/testsuite/gcc.dg/pr96708-positive.c
@@ -42,6 +42,7 @@ int main()
return 0;
}
+/* Note main has one `return 0`. */
/* { dg-final { scan-tree-dump-times "return 0;" 3 "optimized" } } */
/* { dg-final { scan-tree-dump-times "return 1;" 2 "optimized" } } */
/* { dg-final { scan-tree-dump-not { "MAX_EXPR" } "optimized" } } */
diff --git a/gcc/testsuite/gcc.dg/tree-ssa/minmaxcmp-2.c b/gcc/testsuite/gcc.dg/tree-ssa/minmaxcmp-2.c
new file mode 100644
index 00000000000..f64a9253cfb
--- /dev/null
+++ b/gcc/testsuite/gcc.dg/tree-ssa/minmaxcmp-2.c
@@ -0,0 +1,30 @@
+/* { dg-do compile } */
+/* { dg-options "-O2 -fdump-tree-original" } */
+/* PR tree-optimization/111364 */
+
+#define min1(a, b) ((a) < (b) ? (a) : (b))
+#define max1(a, b) ((a) > (b) ? (a) : (b))
+
+int minlt(int a, int b)
+{
+ return min1(a, b) < a; // b < a or a > b
+}
+/* { dg-final { scan-tree-dump "return a > b;|return b < a;" "original" } } */
+
+int minge(int c, int d)
+{
+ return min1(c, d) >= c; // d >= c or c <= d
+}
+/* { dg-final { scan-tree-dump "return c <= d;|return d <= c;" "original" } } */
+
+int maxgt(int e, int f)
+{
+ return max1(e, f) > e; // e > f or f < e
+}
+/* { dg-final { scan-tree-dump "return e < f;|return f > e;" "original" } } */
+
+int maxle(int x, int y)
+{
+ return max1(x, y) <= x; // y <= x or x >= y
+}
+/* { dg-final { scan-tree-dump "return x >= y;|return y <= x;" "original" } } */
--
2.31.1
^ permalink raw reply [flat|nested] 5+ messages in thread
* [PATCH 2/2] MATCH: Move `X <= MAX(X, Y)` before `MIN (X, C1) < C2` pattern
2023-09-12 15:25 [PATCH 1/2] MATCH: [PR111364] Add some more minmax cmp operand simplifications Andrew Pinski
@ 2023-09-12 15:25 ` Andrew Pinski
2023-09-13 6:43 ` Richard Biener
2023-09-13 6:42 ` [PATCH 1/2] MATCH: [PR111364] Add some more minmax cmp operand simplifications Richard Biener
1 sibling, 1 reply; 5+ messages in thread
From: Andrew Pinski @ 2023-09-12 15:25 UTC (permalink / raw)
To: gcc-patches; +Cc: Andrew Pinski
Since matching C1 as C2 here will decrease how much other simplifications
will need to happen to get the final answer.
OK? Bootstrapped and tested on x86_64-linux-gnu.
gcc/ChangeLog:
* match.pd (`X <= MAX(X, Y)`):
Move before `MIN (X, C1) < C2` pattern.
---
gcc/match.pd | 15 ++++++++-------
1 file changed, 8 insertions(+), 7 deletions(-)
diff --git a/gcc/match.pd b/gcc/match.pd
index 36e3da4841b..34b67df784e 100644
--- a/gcc/match.pd
+++ b/gcc/match.pd
@@ -3931,13 +3931,6 @@ DEFINE_INT_AND_FLOAT_ROUND_FN (RINT)
(if (wi::lt_p (wi::to_wide (@1), wi::to_wide (@2),
TYPE_SIGN (TREE_TYPE (@0))))
(cmp @0 @2)))))
-/* MIN (X, C1) < C2 -> X < C2 || C1 < C2 */
-(for minmax (min min max max min min max max )
- cmp (lt le gt ge gt ge lt le )
- comb (bit_ior bit_ior bit_ior bit_ior bit_and bit_and bit_and bit_and)
- (simplify
- (cmp (minmax @0 INTEGER_CST@1) INTEGER_CST@2)
- (comb (cmp @0 @2) (cmp @1 @2))))
/* X <= MAX(X, Y) -> true
X > MAX(X, Y) -> false
@@ -3949,6 +3942,14 @@ DEFINE_INT_AND_FLOAT_ROUND_FN (RINT)
(cmp:c @0 (minmax:c @0 @1))
{ constant_boolean_node (cmp == GE_EXPR || cmp == LE_EXPR, type); } ))
+/* MIN (X, C1) < C2 -> X < C2 || C1 < C2 */
+(for minmax (min min max max min min max max )
+ cmp (lt le gt ge gt ge lt le )
+ comb (bit_ior bit_ior bit_ior bit_ior bit_and bit_and bit_and bit_and)
+ (simplify
+ (cmp (minmax @0 INTEGER_CST@1) INTEGER_CST@2)
+ (comb (cmp @0 @2) (cmp @1 @2))))
+
/* Undo fancy ways of writing max/min or other ?: expressions, like
a - ((a - b) & -(a < b)) and a - (a - b) * (a < b) into (a < b) ? b : a.
People normally use ?: and that is what we actually try to optimize. */
--
2.31.1
^ permalink raw reply [flat|nested] 5+ messages in thread
* Re: [PATCH 1/2] MATCH: [PR111364] Add some more minmax cmp operand simplifications
2023-09-12 15:25 [PATCH 1/2] MATCH: [PR111364] Add some more minmax cmp operand simplifications Andrew Pinski
2023-09-12 15:25 ` [PATCH 2/2] MATCH: Move `X <= MAX(X, Y)` before `MIN (X, C1) < C2` pattern Andrew Pinski
@ 2023-09-13 6:42 ` Richard Biener
2023-09-13 12:17 ` Andrew Pinski
1 sibling, 1 reply; 5+ messages in thread
From: Richard Biener @ 2023-09-13 6:42 UTC (permalink / raw)
To: Andrew Pinski; +Cc: gcc-patches
On Tue, Sep 12, 2023 at 5:31 PM Andrew Pinski via Gcc-patches
<gcc-patches@gcc.gnu.org> wrote:
>
> This adds a few more minmax cmp operand simplifications which were missed before.
> `MIN(a,b) < a` -> `a > b`
> `MIN(a,b) >= a` -> `a <= b`
> `MAX(a,b) > a` -> `a < b`
> `MAX(a,b) <= a` -> `a >= b`
>
> OK? Bootstrapped and tested on x86_64-linux-gnu.
OK. I wonder if any of these are also valid for FP types?
> Note gcc.dg/pr96708-negative.c needed to updated to remove the
> check for MIN/MAX as they have been optimized (correctly) away.
>
> PR tree-optimization/111364
>
> gcc/ChangeLog:
>
> * match.pd (`MIN (X, Y) == X`): Extend
> to min/lt, min/ge, max/gt, max/le.
>
> gcc/testsuite/ChangeLog:
>
> * gcc.c-torture/execute/minmaxcmp-1.c: New test.
> * gcc.dg/tree-ssa/minmaxcmp-2.c: New test.
> * gcc.dg/pr96708-negative.c: Update testcase.
> * gcc.dg/pr96708-positive.c: Add comment about `return 0`.
> ---
> gcc/match.pd | 8 +--
> .../gcc.c-torture/execute/minmaxcmp-1.c | 51 +++++++++++++++++++
> gcc/testsuite/gcc.dg/pr96708-negative.c | 4 +-
> gcc/testsuite/gcc.dg/pr96708-positive.c | 1 +
> gcc/testsuite/gcc.dg/tree-ssa/minmaxcmp-2.c | 30 +++++++++++
> 5 files changed, 89 insertions(+), 5 deletions(-)
> create mode 100644 gcc/testsuite/gcc.c-torture/execute/minmaxcmp-1.c
> create mode 100644 gcc/testsuite/gcc.dg/tree-ssa/minmaxcmp-2.c
>
> diff --git a/gcc/match.pd b/gcc/match.pd
> index 51985c1bad4..36e3da4841b 100644
> --- a/gcc/match.pd
> +++ b/gcc/match.pd
> @@ -3902,9 +3902,11 @@ DEFINE_INT_AND_FLOAT_ROUND_FN (RINT)
> (maxmin @0 (bit_not @1))))
>
> /* MIN (X, Y) == X -> X <= Y */
> -(for minmax (min min max max)
> - cmp (eq ne eq ne )
> - out (le gt ge lt )
> +/* MIN (X, Y) < X -> X > Y */
> +/* MIN (X, Y) >= X -> X <= Y */
> +(for minmax (min min min min max max max max)
> + cmp (eq ne lt ge eq ne gt le )
> + out (le gt gt le ge lt lt ge )
> (simplify
> (cmp:c (minmax:c @0 @1) @0)
> (if (ANY_INTEGRAL_TYPE_P (TREE_TYPE (@0)))
> diff --git a/gcc/testsuite/gcc.c-torture/execute/minmaxcmp-1.c b/gcc/testsuite/gcc.c-torture/execute/minmaxcmp-1.c
> new file mode 100644
> index 00000000000..6705a053768
> --- /dev/null
> +++ b/gcc/testsuite/gcc.c-torture/execute/minmaxcmp-1.c
> @@ -0,0 +1,51 @@
> +#define func(vol, op1, op2) \
> +_Bool op1##_##op2##_##vol (int a, int b) \
> +{ \
> + vol int x = op_##op1(a, b); \
> + return op_##op2(x, a); \
> +}
> +
> +#define op_lt(a, b) ((a) < (b))
> +#define op_le(a, b) ((a) <= (b))
> +#define op_eq(a, b) ((a) == (b))
> +#define op_ne(a, b) ((a) != (b))
> +#define op_gt(a, b) ((a) > (b))
> +#define op_ge(a, b) ((a) >= (b))
> +#define op_min(a, b) ((a) < (b) ? (a) : (b))
> +#define op_max(a, b) ((a) > (b) ? (a) : (b))
> +
> +
> +#define funcs(a) \
> + a(min,lt) \
> + a(max,lt) \
> + a(min,gt) \
> + a(max,gt) \
> + a(min,le) \
> + a(max,le) \
> + a(min,ge) \
> + a(max,ge) \
> + a(min,ne) \
> + a(max,ne) \
> + a(min,eq) \
> + a(max,eq)
> +
> +#define funcs1(a,b) \
> +func(,a,b) \
> +func(volatile,a,b)
> +
> +funcs(funcs1)
> +
> +#define test(op1,op2) \
> +do { \
> + if (op1##_##op2##_(x,y) != op1##_##op2##_volatile(x,y)) \
> + __builtin_abort(); \
> +} while(0);
> +
> +int main()
> +{
> + for(int x = -10; x < 10; x++)
> + for(int y = -10; y < 10; y++)
> + {
> + funcs(test)
> + }
> +}
> diff --git a/gcc/testsuite/gcc.dg/pr96708-negative.c b/gcc/testsuite/gcc.dg/pr96708-negative.c
> index 91964d3b971..c9c1aa85558 100644
> --- a/gcc/testsuite/gcc.dg/pr96708-negative.c
> +++ b/gcc/testsuite/gcc.dg/pr96708-negative.c
> @@ -42,7 +42,7 @@ int main()
> return 0;
> }
>
> -/* { dg-final { scan-tree-dump-times "MAX_EXPR" 2 "optimized" } } */
> -/* { dg-final { scan-tree-dump-times "MIN_EXPR" 2 "optimized" } } */
> +/* Even though test[1-4] originally has MIN/MAX, those can be optimized away
> + into just comparing a and b arguments. */
> /* { dg-final { scan-tree-dump-times "return 0;" 1 "optimized" } } */
> /* { dg-final { scan-tree-dump-not { "return 1;" } "optimized" } } */
> diff --git a/gcc/testsuite/gcc.dg/pr96708-positive.c b/gcc/testsuite/gcc.dg/pr96708-positive.c
> index 65af85344b6..12c5fedfd30 100644
> --- a/gcc/testsuite/gcc.dg/pr96708-positive.c
> +++ b/gcc/testsuite/gcc.dg/pr96708-positive.c
> @@ -42,6 +42,7 @@ int main()
> return 0;
> }
>
> +/* Note main has one `return 0`. */
> /* { dg-final { scan-tree-dump-times "return 0;" 3 "optimized" } } */
> /* { dg-final { scan-tree-dump-times "return 1;" 2 "optimized" } } */
> /* { dg-final { scan-tree-dump-not { "MAX_EXPR" } "optimized" } } */
> diff --git a/gcc/testsuite/gcc.dg/tree-ssa/minmaxcmp-2.c b/gcc/testsuite/gcc.dg/tree-ssa/minmaxcmp-2.c
> new file mode 100644
> index 00000000000..f64a9253cfb
> --- /dev/null
> +++ b/gcc/testsuite/gcc.dg/tree-ssa/minmaxcmp-2.c
> @@ -0,0 +1,30 @@
> +/* { dg-do compile } */
> +/* { dg-options "-O2 -fdump-tree-original" } */
> +/* PR tree-optimization/111364 */
> +
> +#define min1(a, b) ((a) < (b) ? (a) : (b))
> +#define max1(a, b) ((a) > (b) ? (a) : (b))
> +
> +int minlt(int a, int b)
> +{
> + return min1(a, b) < a; // b < a or a > b
> +}
> +/* { dg-final { scan-tree-dump "return a > b;|return b < a;" "original" } } */
> +
> +int minge(int c, int d)
> +{
> + return min1(c, d) >= c; // d >= c or c <= d
> +}
> +/* { dg-final { scan-tree-dump "return c <= d;|return d <= c;" "original" } } */
> +
> +int maxgt(int e, int f)
> +{
> + return max1(e, f) > e; // e > f or f < e
> +}
> +/* { dg-final { scan-tree-dump "return e < f;|return f > e;" "original" } } */
> +
> +int maxle(int x, int y)
> +{
> + return max1(x, y) <= x; // y <= x or x >= y
> +}
> +/* { dg-final { scan-tree-dump "return x >= y;|return y <= x;" "original" } } */
> --
> 2.31.1
>
^ permalink raw reply [flat|nested] 5+ messages in thread
* Re: [PATCH 2/2] MATCH: Move `X <= MAX(X, Y)` before `MIN (X, C1) < C2` pattern
2023-09-12 15:25 ` [PATCH 2/2] MATCH: Move `X <= MAX(X, Y)` before `MIN (X, C1) < C2` pattern Andrew Pinski
@ 2023-09-13 6:43 ` Richard Biener
0 siblings, 0 replies; 5+ messages in thread
From: Richard Biener @ 2023-09-13 6:43 UTC (permalink / raw)
To: Andrew Pinski; +Cc: gcc-patches
On Tue, Sep 12, 2023 at 5:41 PM Andrew Pinski via Gcc-patches
<gcc-patches@gcc.gnu.org> wrote:
>
> Since matching C1 as C2 here will decrease how much other simplifications
> will need to happen to get the final answer.
>
> OK? Bootstrapped and tested on x86_64-linux-gnu.
OK
Richard.
> gcc/ChangeLog:
>
> * match.pd (`X <= MAX(X, Y)`):
> Move before `MIN (X, C1) < C2` pattern.
> ---
> gcc/match.pd | 15 ++++++++-------
> 1 file changed, 8 insertions(+), 7 deletions(-)
>
> diff --git a/gcc/match.pd b/gcc/match.pd
> index 36e3da4841b..34b67df784e 100644
> --- a/gcc/match.pd
> +++ b/gcc/match.pd
> @@ -3931,13 +3931,6 @@ DEFINE_INT_AND_FLOAT_ROUND_FN (RINT)
> (if (wi::lt_p (wi::to_wide (@1), wi::to_wide (@2),
> TYPE_SIGN (TREE_TYPE (@0))))
> (cmp @0 @2)))))
> -/* MIN (X, C1) < C2 -> X < C2 || C1 < C2 */
> -(for minmax (min min max max min min max max )
> - cmp (lt le gt ge gt ge lt le )
> - comb (bit_ior bit_ior bit_ior bit_ior bit_and bit_and bit_and bit_and)
> - (simplify
> - (cmp (minmax @0 INTEGER_CST@1) INTEGER_CST@2)
> - (comb (cmp @0 @2) (cmp @1 @2))))
>
> /* X <= MAX(X, Y) -> true
> X > MAX(X, Y) -> false
> @@ -3949,6 +3942,14 @@ DEFINE_INT_AND_FLOAT_ROUND_FN (RINT)
> (cmp:c @0 (minmax:c @0 @1))
> { constant_boolean_node (cmp == GE_EXPR || cmp == LE_EXPR, type); } ))
>
> +/* MIN (X, C1) < C2 -> X < C2 || C1 < C2 */
> +(for minmax (min min max max min min max max )
> + cmp (lt le gt ge gt ge lt le )
> + comb (bit_ior bit_ior bit_ior bit_ior bit_and bit_and bit_and bit_and)
> + (simplify
> + (cmp (minmax @0 INTEGER_CST@1) INTEGER_CST@2)
> + (comb (cmp @0 @2) (cmp @1 @2))))
> +
> /* Undo fancy ways of writing max/min or other ?: expressions, like
> a - ((a - b) & -(a < b)) and a - (a - b) * (a < b) into (a < b) ? b : a.
> People normally use ?: and that is what we actually try to optimize. */
> --
> 2.31.1
>
^ permalink raw reply [flat|nested] 5+ messages in thread
* Re: [PATCH 1/2] MATCH: [PR111364] Add some more minmax cmp operand simplifications
2023-09-13 6:42 ` [PATCH 1/2] MATCH: [PR111364] Add some more minmax cmp operand simplifications Richard Biener
@ 2023-09-13 12:17 ` Andrew Pinski
0 siblings, 0 replies; 5+ messages in thread
From: Andrew Pinski @ 2023-09-13 12:17 UTC (permalink / raw)
To: Richard Biener; +Cc: Andrew Pinski, gcc-patches
On Tue, Sep 12, 2023 at 11:45 PM Richard Biener via Gcc-patches
<gcc-patches@gcc.gnu.org> wrote:
>
> On Tue, Sep 12, 2023 at 5:31 PM Andrew Pinski via Gcc-patches
> <gcc-patches@gcc.gnu.org> wrote:
> >
> > This adds a few more minmax cmp operand simplifications which were missed before.
> > `MIN(a,b) < a` -> `a > b`
> > `MIN(a,b) >= a` -> `a <= b`
> > `MAX(a,b) > a` -> `a < b`
> > `MAX(a,b) <= a` -> `a >= b`
> >
> > OK? Bootstrapped and tested on x86_64-linux-gnu.
>
> OK. I wonder if any of these are also valid for FP types?
I was thinking about that too. I will look into that later this week.
Thanks,
Andrew
>
> > Note gcc.dg/pr96708-negative.c needed to updated to remove the
> > check for MIN/MAX as they have been optimized (correctly) away.
> >
> > PR tree-optimization/111364
> >
> > gcc/ChangeLog:
> >
> > * match.pd (`MIN (X, Y) == X`): Extend
> > to min/lt, min/ge, max/gt, max/le.
> >
> > gcc/testsuite/ChangeLog:
> >
> > * gcc.c-torture/execute/minmaxcmp-1.c: New test.
> > * gcc.dg/tree-ssa/minmaxcmp-2.c: New test.
> > * gcc.dg/pr96708-negative.c: Update testcase.
> > * gcc.dg/pr96708-positive.c: Add comment about `return 0`.
> > ---
> > gcc/match.pd | 8 +--
> > .../gcc.c-torture/execute/minmaxcmp-1.c | 51 +++++++++++++++++++
> > gcc/testsuite/gcc.dg/pr96708-negative.c | 4 +-
> > gcc/testsuite/gcc.dg/pr96708-positive.c | 1 +
> > gcc/testsuite/gcc.dg/tree-ssa/minmaxcmp-2.c | 30 +++++++++++
> > 5 files changed, 89 insertions(+), 5 deletions(-)
> > create mode 100644 gcc/testsuite/gcc.c-torture/execute/minmaxcmp-1.c
> > create mode 100644 gcc/testsuite/gcc.dg/tree-ssa/minmaxcmp-2.c
> >
> > diff --git a/gcc/match.pd b/gcc/match.pd
> > index 51985c1bad4..36e3da4841b 100644
> > --- a/gcc/match.pd
> > +++ b/gcc/match.pd
> > @@ -3902,9 +3902,11 @@ DEFINE_INT_AND_FLOAT_ROUND_FN (RINT)
> > (maxmin @0 (bit_not @1))))
> >
> > /* MIN (X, Y) == X -> X <= Y */
> > -(for minmax (min min max max)
> > - cmp (eq ne eq ne )
> > - out (le gt ge lt )
> > +/* MIN (X, Y) < X -> X > Y */
> > +/* MIN (X, Y) >= X -> X <= Y */
> > +(for minmax (min min min min max max max max)
> > + cmp (eq ne lt ge eq ne gt le )
> > + out (le gt gt le ge lt lt ge )
> > (simplify
> > (cmp:c (minmax:c @0 @1) @0)
> > (if (ANY_INTEGRAL_TYPE_P (TREE_TYPE (@0)))
> > diff --git a/gcc/testsuite/gcc.c-torture/execute/minmaxcmp-1.c b/gcc/testsuite/gcc.c-torture/execute/minmaxcmp-1.c
> > new file mode 100644
> > index 00000000000..6705a053768
> > --- /dev/null
> > +++ b/gcc/testsuite/gcc.c-torture/execute/minmaxcmp-1.c
> > @@ -0,0 +1,51 @@
> > +#define func(vol, op1, op2) \
> > +_Bool op1##_##op2##_##vol (int a, int b) \
> > +{ \
> > + vol int x = op_##op1(a, b); \
> > + return op_##op2(x, a); \
> > +}
> > +
> > +#define op_lt(a, b) ((a) < (b))
> > +#define op_le(a, b) ((a) <= (b))
> > +#define op_eq(a, b) ((a) == (b))
> > +#define op_ne(a, b) ((a) != (b))
> > +#define op_gt(a, b) ((a) > (b))
> > +#define op_ge(a, b) ((a) >= (b))
> > +#define op_min(a, b) ((a) < (b) ? (a) : (b))
> > +#define op_max(a, b) ((a) > (b) ? (a) : (b))
> > +
> > +
> > +#define funcs(a) \
> > + a(min,lt) \
> > + a(max,lt) \
> > + a(min,gt) \
> > + a(max,gt) \
> > + a(min,le) \
> > + a(max,le) \
> > + a(min,ge) \
> > + a(max,ge) \
> > + a(min,ne) \
> > + a(max,ne) \
> > + a(min,eq) \
> > + a(max,eq)
> > +
> > +#define funcs1(a,b) \
> > +func(,a,b) \
> > +func(volatile,a,b)
> > +
> > +funcs(funcs1)
> > +
> > +#define test(op1,op2) \
> > +do { \
> > + if (op1##_##op2##_(x,y) != op1##_##op2##_volatile(x,y)) \
> > + __builtin_abort(); \
> > +} while(0);
> > +
> > +int main()
> > +{
> > + for(int x = -10; x < 10; x++)
> > + for(int y = -10; y < 10; y++)
> > + {
> > + funcs(test)
> > + }
> > +}
> > diff --git a/gcc/testsuite/gcc.dg/pr96708-negative.c b/gcc/testsuite/gcc.dg/pr96708-negative.c
> > index 91964d3b971..c9c1aa85558 100644
> > --- a/gcc/testsuite/gcc.dg/pr96708-negative.c
> > +++ b/gcc/testsuite/gcc.dg/pr96708-negative.c
> > @@ -42,7 +42,7 @@ int main()
> > return 0;
> > }
> >
> > -/* { dg-final { scan-tree-dump-times "MAX_EXPR" 2 "optimized" } } */
> > -/* { dg-final { scan-tree-dump-times "MIN_EXPR" 2 "optimized" } } */
> > +/* Even though test[1-4] originally has MIN/MAX, those can be optimized away
> > + into just comparing a and b arguments. */
> > /* { dg-final { scan-tree-dump-times "return 0;" 1 "optimized" } } */
> > /* { dg-final { scan-tree-dump-not { "return 1;" } "optimized" } } */
> > diff --git a/gcc/testsuite/gcc.dg/pr96708-positive.c b/gcc/testsuite/gcc.dg/pr96708-positive.c
> > index 65af85344b6..12c5fedfd30 100644
> > --- a/gcc/testsuite/gcc.dg/pr96708-positive.c
> > +++ b/gcc/testsuite/gcc.dg/pr96708-positive.c
> > @@ -42,6 +42,7 @@ int main()
> > return 0;
> > }
> >
> > +/* Note main has one `return 0`. */
> > /* { dg-final { scan-tree-dump-times "return 0;" 3 "optimized" } } */
> > /* { dg-final { scan-tree-dump-times "return 1;" 2 "optimized" } } */
> > /* { dg-final { scan-tree-dump-not { "MAX_EXPR" } "optimized" } } */
> > diff --git a/gcc/testsuite/gcc.dg/tree-ssa/minmaxcmp-2.c b/gcc/testsuite/gcc.dg/tree-ssa/minmaxcmp-2.c
> > new file mode 100644
> > index 00000000000..f64a9253cfb
> > --- /dev/null
> > +++ b/gcc/testsuite/gcc.dg/tree-ssa/minmaxcmp-2.c
> > @@ -0,0 +1,30 @@
> > +/* { dg-do compile } */
> > +/* { dg-options "-O2 -fdump-tree-original" } */
> > +/* PR tree-optimization/111364 */
> > +
> > +#define min1(a, b) ((a) < (b) ? (a) : (b))
> > +#define max1(a, b) ((a) > (b) ? (a) : (b))
> > +
> > +int minlt(int a, int b)
> > +{
> > + return min1(a, b) < a; // b < a or a > b
> > +}
> > +/* { dg-final { scan-tree-dump "return a > b;|return b < a;" "original" } } */
> > +
> > +int minge(int c, int d)
> > +{
> > + return min1(c, d) >= c; // d >= c or c <= d
> > +}
> > +/* { dg-final { scan-tree-dump "return c <= d;|return d <= c;" "original" } } */
> > +
> > +int maxgt(int e, int f)
> > +{
> > + return max1(e, f) > e; // e > f or f < e
> > +}
> > +/* { dg-final { scan-tree-dump "return e < f;|return f > e;" "original" } } */
> > +
> > +int maxle(int x, int y)
> > +{
> > + return max1(x, y) <= x; // y <= x or x >= y
> > +}
> > +/* { dg-final { scan-tree-dump "return x >= y;|return y <= x;" "original" } } */
> > --
> > 2.31.1
> >
^ permalink raw reply [flat|nested] 5+ messages in thread
end of thread, other threads:[~2023-09-13 12:17 UTC | newest]
Thread overview: 5+ messages (download: mbox.gz / follow: Atom feed)
-- links below jump to the message on this page --
2023-09-12 15:25 [PATCH 1/2] MATCH: [PR111364] Add some more minmax cmp operand simplifications Andrew Pinski
2023-09-12 15:25 ` [PATCH 2/2] MATCH: Move `X <= MAX(X, Y)` before `MIN (X, C1) < C2` pattern Andrew Pinski
2023-09-13 6:43 ` Richard Biener
2023-09-13 6:42 ` [PATCH 1/2] MATCH: [PR111364] Add some more minmax cmp operand simplifications Richard Biener
2023-09-13 12:17 ` Andrew Pinski
This is a public inbox, see mirroring instructions
for how to clone and mirror all data and code used for this inbox;
as well as URLs for read-only IMAP folder(s) and NNTP newsgroup(s).