aboutsummaryrefslogtreecommitdiff
path: root/gcc
diff options
context:
space:
mode:
authorRichard Sandiford <richard.sandiford@arm.com>2024-10-24 14:22:34 +0100
committerRichard Sandiford <richard.sandiford@arm.com>2024-10-24 14:22:34 +0100
commitb02503d3fe79778ccc149b0abbc736f7972603da (patch)
tree488f7ecac9b362388ec22243000eca0fc6754332 /gcc
parentaf19e46c88fd75e31127cde239b8f28d8f9c4040 (diff)
downloadgcc-b02503d3fe79778ccc149b0abbc736f7972603da.zip
gcc-b02503d3fe79778ccc149b0abbc736f7972603da.tar.gz
gcc-b02503d3fe79778ccc149b0abbc736f7972603da.tar.bz2
Record nonzero bits in the irange_bitmask of POLY_INT_CSTs
At the moment, ranger punts entirely on POLY_INT_CSTs. Numerical ranges are a bit difficult, unless we do start modelling bounds on the indeterminates. But we can at least track the nonzero bits. gcc/ * value-query.cc (range_query::get_tree_range): Use get_nonzero_bits to populate the irange_bitmask of a POLY_INT_CST. gcc/testsuite/ * gcc.target/aarch64/sve/cnt_fold_6.c: New test.
Diffstat (limited to 'gcc')
-rw-r--r--gcc/testsuite/gcc.target/aarch64/sve/cnt_fold_6.c75
-rw-r--r--gcc/value-query.cc7
2 files changed, 82 insertions, 0 deletions
diff --git a/gcc/testsuite/gcc.target/aarch64/sve/cnt_fold_6.c b/gcc/testsuite/gcc.target/aarch64/sve/cnt_fold_6.c
new file mode 100644
index 0000000..9d9e1ca
--- /dev/null
+++ b/gcc/testsuite/gcc.target/aarch64/sve/cnt_fold_6.c
@@ -0,0 +1,75 @@
+/* { dg-do compile } */
+/* { dg-options "-O2" } */
+/* { dg-final { check-function-bodies "**" "" } } */
+
+#include <arm_sve.h>
+
+/*
+** f1:
+** ...
+** cntb (x[0-9]+)
+** ...
+** add x[0-9]+, \1, #?16
+** ...
+** csel [^\n]+
+** ret
+*/
+uint64_t
+f1 (int x)
+{
+ uint64_t y = x ? svcnth () : svcnth () + 8;
+ y >>= 3;
+ y <<= 4;
+ return y;
+}
+
+/*
+** f2:
+** ...
+** (?:and|[al]sr) [^\n]+
+** ...
+** ret
+*/
+uint64_t
+f2 (int x)
+{
+ uint64_t y = x ? svcnth () : svcnth () + 8;
+ y >>= 4;
+ y <<= 5;
+ return y;
+}
+
+/*
+** f3:
+** ...
+** cntw (x[0-9]+)
+** ...
+** add x[0-9]+, \1, #?16
+** ...
+** csel [^\n]+
+** ret
+*/
+uint64_t
+f3 (int x)
+{
+ uint64_t y = x ? svcntd () : svcntd () + 8;
+ y >>= 1;
+ y <<= 2;
+ return y;
+}
+
+/*
+** f4:
+** ...
+** (?:and|[al]sr) [^\n]+
+** ...
+** ret
+*/
+uint64_t
+f4 (int x)
+{
+ uint64_t y = x ? svcntd () : svcntd () + 8;
+ y >>= 2;
+ y <<= 3;
+ return y;
+}
diff --git a/gcc/value-query.cc b/gcc/value-query.cc
index cac2cb5..34499da 100644
--- a/gcc/value-query.cc
+++ b/gcc/value-query.cc
@@ -375,6 +375,13 @@ range_query::get_tree_range (vrange &r, tree expr, gimple *stmt,
}
default:
+ if (POLY_INT_CST_P (expr))
+ {
+ unsigned int precision = TYPE_PRECISION (type);
+ r.set_varying (type);
+ r.update_bitmask ({ wi::zero (precision), get_nonzero_bits (expr) });
+ return true;
+ }
break;
}
if (BINARY_CLASS_P (expr) || COMPARISON_CLASS_P (expr))