forked from OSchip/llvm-project
[SVE] Fix invalid use of VectorType::getNumElements() in PatternMatch
Summary: Update cst_pred_ty to only work on FixedVectorType. It operates on integers and integer vectors, and returns true if the predicate returns true for all elements of the vector. This operation is not possible on scalable vectors. Make this behavior explicit in the code and document the fact that it only tests fixed width vectors. Identified by test LLVM.Transforms/InstCombine::nsw.ll Reviewers: efriedma, c-rhodes, david-arm, spatel Reviewed By: david-arm Subscribers: tschuett, rkruppe, psnobl, llvm-commits Tags: #llvm Differential Revision: https://reviews.llvm.org/D79195
This commit is contained in:
parent
78769923fe
commit
0ee7b7e3f1
|
@ -262,20 +262,20 @@ template <int64_t Val> inline constantint_match<Val> m_ConstantInt() {
|
||||||
return constantint_match<Val>();
|
return constantint_match<Val>();
|
||||||
}
|
}
|
||||||
|
|
||||||
/// This helper class is used to match scalar and vector integer constants that
|
/// This helper class is used to match scalar and fixed width vector integer
|
||||||
/// satisfy a specified predicate.
|
/// constants that satisfy a specified predicate.
|
||||||
/// For vector constants, undefined elements are ignored.
|
/// For vector constants, undefined elements are ignored.
|
||||||
template <typename Predicate> struct cst_pred_ty : public Predicate {
|
template <typename Predicate> struct cst_pred_ty : public Predicate {
|
||||||
template <typename ITy> bool match(ITy *V) {
|
template <typename ITy> bool match(ITy *V) {
|
||||||
if (const auto *CI = dyn_cast<ConstantInt>(V))
|
if (const auto *CI = dyn_cast<ConstantInt>(V))
|
||||||
return this->isValue(CI->getValue());
|
return this->isValue(CI->getValue());
|
||||||
if (V->getType()->isVectorTy()) {
|
if (const auto *FVTy = dyn_cast<FixedVectorType>(V->getType())) {
|
||||||
if (const auto *C = dyn_cast<Constant>(V)) {
|
if (const auto *C = dyn_cast<Constant>(V)) {
|
||||||
if (const auto *CI = dyn_cast_or_null<ConstantInt>(C->getSplatValue()))
|
if (const auto *CI = dyn_cast_or_null<ConstantInt>(C->getSplatValue()))
|
||||||
return this->isValue(CI->getValue());
|
return this->isValue(CI->getValue());
|
||||||
|
|
||||||
// Non-splat vector constant: check each element for a match.
|
// Non-splat vector constant: check each element for a match.
|
||||||
unsigned NumElts = cast<VectorType>(V->getType())->getNumElements();
|
unsigned NumElts = FVTy->getNumElements();
|
||||||
assert(NumElts != 0 && "Constant vector with no elements?");
|
assert(NumElts != 0 && "Constant vector with no elements?");
|
||||||
bool HasNonUndefElements = false;
|
bool HasNonUndefElements = false;
|
||||||
for (unsigned i = 0; i != NumElts; ++i) {
|
for (unsigned i = 0; i != NumElts; ++i) {
|
||||||
|
@ -462,6 +462,7 @@ inline cst_pred_ty<is_zero_int> m_ZeroInt() {
|
||||||
struct is_zero {
|
struct is_zero {
|
||||||
template <typename ITy> bool match(ITy *V) {
|
template <typename ITy> bool match(ITy *V) {
|
||||||
auto *C = dyn_cast<Constant>(V);
|
auto *C = dyn_cast<Constant>(V);
|
||||||
|
// FIXME: this should be able to do something for scalable vectors
|
||||||
return C && (C->isNullValue() || cst_pred_ty<is_zero_int>().match(C));
|
return C && (C->isNullValue() || cst_pred_ty<is_zero_int>().match(C));
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
Loading…
Reference in New Issue