diff options
author | Richard Sandiford <richard.sandiford@arm.com> | 2024-10-24 14:22:34 +0100 |
---|---|---|
committer | Richard Sandiford <richard.sandiford@arm.com> | 2024-10-24 14:22:34 +0100 |
commit | b02503d3fe79778ccc149b0abbc736f7972603da (patch) | |
tree | 488f7ecac9b362388ec22243000eca0fc6754332 /gcc | |
parent | af19e46c88fd75e31127cde239b8f28d8f9c4040 (diff) | |
download | gcc-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.c | 75 | ||||
-rw-r--r-- | gcc/value-query.cc | 7 |
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)) |