Add a pattern for `(trunc)copysign ((extend)x, CST)`. Only the sign of
CST matters, not its value, so it can be simplified to
`copysign (x, -1.0/1.0)` depending on the sign of CST.
PR tree-optimization/112472
gcc/ChangeLog:
* match.pd ((trunc)copysign ((extend)x, CST) --> copysign (x,
-1.0/1.0)):
New pattern.
gcc/testsuite/ChangeLog:
* gcc.dg/tree-ssa/pr112472.c: New test.
Signed-off-by: Eikansh Gupta <[email protected]>
---
gcc/match.pd | 14 +++++++++++++-
gcc/testsuite/gcc.dg/tree-ssa/pr112472.c | 15 +++++++++++++++
2 files changed, 28 insertions(+), 1 deletion(-)
create mode 100644 gcc/testsuite/gcc.dg/tree-ssa/pr112472.c
diff --git a/gcc/match.pd b/gcc/match.pd
index ddf3b61638c..7edc73945fb 100644
--- a/gcc/match.pd
+++ b/gcc/match.pd
@@ -9343,7 +9343,19 @@ DEFINE_INT_AND_FLOAT_ROUND_FN (RINT)
&& TYPE_PRECISION (type) < TYPE_PRECISION (TREE_TYPE (@2))
&& direct_internal_fn_supported_p (IFN_COPYSIGN,
type, OPTIMIZE_FOR_BOTH))
- (IFN_COPYSIGN @0 @1))))
+ (IFN_COPYSIGN @0 @1)))
+ /* Simplify (trunc)copysign ((extend)x, CST) to copysign (x, -1.0/1.0). */
+ (simplify
+ (convert (copysigns (convert@2 @0) REAL_CST@1))
+ (if (optimize
+ && !HONOR_SNANS (@2)
+ && types_match (type, TREE_TYPE (@0))
+ && TYPE_PRECISION (type) < TYPE_PRECISION (TREE_TYPE (@2))
+ && direct_internal_fn_supported_p (IFN_COPYSIGN,
+ type, OPTIMIZE_FOR_BOTH))
+ (if (REAL_VALUE_NEGATIVE (TREE_REAL_CST (@1)))
+ (IFN_COPYSIGN @0 { build_minus_one_cst (type); })
+ (IFN_COPYSIGN @0 { build_one_cst (type); })))))
(for froms (BUILT_IN_FMAF BUILT_IN_FMA BUILT_IN_FMAL)
tos (IFN_FMA IFN_FMA IFN_FMA)
diff --git a/gcc/testsuite/gcc.dg/tree-ssa/pr112472.c
b/gcc/testsuite/gcc.dg/tree-ssa/pr112472.c
new file mode 100644
index 00000000000..30ef7636a50
--- /dev/null
+++ b/gcc/testsuite/gcc.dg/tree-ssa/pr112472.c
@@ -0,0 +1,15 @@
+/* PR tree-optimization/112472 */
+/* { dg-do compile } */
+/* { dg-options "-O1 -fdump-tree-optimized" } */
+
+/* (trunc)copysign ((extend)a, CST) with a negative CST should be
+ simplified to .COPYSIGN (a, -1.0e+0). */
+float f(float a)
+{
+ return (float)__builtin_copysign(a, -3.0);
+}
+
+/* { dg-final { scan-tree-dump-not "= __builtin_copysign" "optimized" } } */
+/* { dg-final { scan-tree-dump-not " double " "optimized" { target
ifn_copysign } } } */
+/* { dg-final { scan-tree-dump-times ".COPYSIGN" 1 "optimized" { target
ifn_copysign } } } */
+/* { dg-final { scan-tree-dump-times "-1.0e\\+0" 1 "optimized" { target
ifn_copysign } } } */
--
2.34.1
>From bcf5f5a3836a3c0841f70dc98b964003fb3d1c89 Mon Sep 17 00:00:00 2001
From: Eikansh Gupta <[email protected]>
Date: Mon, 15 Jun 2026 12:19:45 +0530
Subject: [PATCH v3 2/2] MATCH: Simplify `(trunc)abs ((extend)x)` to `abs (x)`
[PR112472]
When the second argument of copysign is a non-negative constant, the
inner copysign is canonicalized to abs before the copysign pattern can
match, leaving (trunc)abs ((extend)x) with a redundant extend/truncate.
Add a pattern so the non-negative-constant copysign case is fully optimized too.
PR tree-optimization/112472
gcc/ChangeLog:
* match.pd ((trunc)abs (extend x) --> abs (x)): New pattern.
gcc/testsuite/ChangeLog:
* gcc.dg/tree-ssa/pr112472.c: Add non-negative constant case.
Signed-off-by: Eikansh Gupta <[email protected]>
---
gcc/match.pd | 9 +++++++++
gcc/testsuite/gcc.dg/tree-ssa/pr112472.c | 9 +++++++++
2 files changed, 18 insertions(+)
diff --git a/gcc/match.pd b/gcc/match.pd
index 7edc73945fb..d627aa4728e 100644
--- a/gcc/match.pd
+++ b/gcc/match.pd
@@ -9357,6 +9357,15 @@ DEFINE_INT_AND_FLOAT_ROUND_FN (RINT)
(IFN_COPYSIGN @0 { build_minus_one_cst (type); })
(IFN_COPYSIGN @0 { build_one_cst (type); })))))
+/* (trunc)abs (extend x) --> abs (x) */
+(simplify
+ (convert (abs (convert@1 @0)))
+ (if (optimize
+ && !HONOR_SNANS (@1)
+ && types_match (type, TREE_TYPE (@0))
+ && TYPE_PRECISION (type) < TYPE_PRECISION (TREE_TYPE (@1)))
+ (abs @0)))
+
(for froms (BUILT_IN_FMAF BUILT_IN_FMA BUILT_IN_FMAL)
tos (IFN_FMA IFN_FMA IFN_FMA)
(simplify
diff --git a/gcc/testsuite/gcc.dg/tree-ssa/pr112472.c
b/gcc/testsuite/gcc.dg/tree-ssa/pr112472.c
index 30ef7636a50..6838c73c77e 100644
--- a/gcc/testsuite/gcc.dg/tree-ssa/pr112472.c
+++ b/gcc/testsuite/gcc.dg/tree-ssa/pr112472.c
@@ -9,7 +9,16 @@ float f(float a)
return (float)__builtin_copysign(a, -3.0);
}
+/* With a non-negative CST, copysign is canonicalized to abs, so this
+ becomes (float)abs((double)a) and is then simplified to abs(a),
+ dropping the wider type. */
+float f2(float a)
+{
+ return (float)__builtin_copysign(a, 5.0);
+}
+
/* { dg-final { scan-tree-dump-not "= __builtin_copysign" "optimized" } } */
/* { dg-final { scan-tree-dump-not " double " "optimized" { target
ifn_copysign } } } */
/* { dg-final { scan-tree-dump-times ".COPYSIGN" 1 "optimized" { target
ifn_copysign } } } */
/* { dg-final { scan-tree-dump-times "-1.0e\\+0" 1 "optimized" { target
ifn_copysign } } } */
+/* { dg-final { scan-tree-dump-times " ABS_EXPR " 1 "optimized" { target
ifn_copysign } } } */
--
2.34.1