The pattern ends up using the argument more than one time which
isn't supported. When FFS directly maps to CTZ + 1 it works though.
Bootstrap and regtest running on x86_64-unknown-linux-gnu.
PR tree-optimization/123038
* tree-vect-patterns.cc (vect_recog_ctz_ffs_pattern): Reject
pattern for reductions when the call argument is used multiple
times.
* gcc.dg/vect/pr123038.c: New testcase.
---
gcc/testsuite/gcc.dg/vect/pr123038.c | 8 ++++++++
gcc/tree-vect-patterns.cc | 14 ++++++++++++++
2 files changed, 22 insertions(+)
create mode 100644 gcc/testsuite/gcc.dg/vect/pr123038.c
diff --git a/gcc/testsuite/gcc.dg/vect/pr123038.c
b/gcc/testsuite/gcc.dg/vect/pr123038.c
new file mode 100644
index 00000000000..bca831f9c08
--- /dev/null
+++ b/gcc/testsuite/gcc.dg/vect/pr123038.c
@@ -0,0 +1,8 @@
+/* { dg-do compile } */
+
+unsigned char f(int b)
+{
+ for (int a = 0; a < 10; a += 1)
+ b = __builtin_ffs(b);
+ return b;
+}
diff --git a/gcc/tree-vect-patterns.cc b/gcc/tree-vect-patterns.cc
index 555986b4fec..af64cb84e41 100644
--- a/gcc/tree-vect-patterns.cc
+++ b/gcc/tree-vect-patterns.cc
@@ -1914,6 +1914,9 @@ vect_recog_ctz_ffs_pattern (vec_info *vinfo,
stmt_vec_info stmt_vinfo,
&& val_new == prec)
|| (ifnnew == IFN_POPCOUNT && ifn == IFN_CTZ))
{
+ if (vect_is_reduction (stmt_vinfo))
+ return NULL;
+
/* .CTZ (X) = PREC - .CLZ ((X - 1) & ~X)
.CTZ (X) = .POPCOUNT ((X - 1) & ~X). */
if (ifnnew == IFN_CLZ)
@@ -1952,6 +1955,9 @@ vect_recog_ctz_ffs_pattern (vec_info *vinfo,
stmt_vec_info stmt_vinfo,
}
else if (ifnnew == IFN_CLZ)
{
+ if (vect_is_reduction (stmt_vinfo))
+ return NULL;
+
/* .CTZ (X) = (PREC - 1) - .CLZ (X & -X)
.FFS (X) = PREC - .CLZ (X & -X). */
sub = prec - (ifn == IFN_CTZ);
@@ -1971,6 +1977,9 @@ vect_recog_ctz_ffs_pattern (vec_info *vinfo,
stmt_vec_info stmt_vinfo,
}
else if (ifnnew == IFN_POPCOUNT)
{
+ if (vect_is_reduction (stmt_vinfo))
+ return NULL;
+
/* .CTZ (X) = PREC - .POPCOUNT (X | -X)
.FFS (X) = (PREC + 1) - .POPCOUNT (X | -X). */
sub = prec + (ifn == IFN_FFS);
@@ -1993,6 +2002,11 @@ vect_recog_ctz_ffs_pattern (vec_info *vinfo,
stmt_vec_info stmt_vinfo,
/* .FFS (X) = .CTZ (X) + 1. */
add = 1;
val_cmp++;
+
+ if (vect_is_reduction (stmt_vinfo)
+ && defined_at_zero
+ && (!defined_at_zero_new || val != val_cmp))
+ return NULL;
}
/* Create B = .IFNNEW (A). */
--
2.51.0