| //==-- OverflowInstAnalysis.cpp - Utils to fold overflow insts ----*- C++ -*-=// |
| // |
| // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions. |
| // See https://llvm.org/LICENSE.txt for license information. |
| // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception |
| // |
| //===----------------------------------------------------------------------===// |
| // |
| // This file holds routines to help analyse overflow instructions |
| // and fold them into constants or other overflow instructions |
| // |
| //===----------------------------------------------------------------------===// |
| |
| #include "llvm/Analysis/OverflowInstAnalysis.h" |
| #include "llvm/IR/Constants.h" |
| #include "llvm/IR/Instructions.h" |
| #include "llvm/IR/PatternMatch.h" |
| |
| using namespace llvm; |
| using namespace llvm::PatternMatch; |
| |
| bool llvm::isCheckForZeroAndMulWithOverflow(Value *Op0, Value *Op1, bool IsAnd, |
| Use *&Y) { |
| ICmpInst::Predicate Pred; |
| Value *X, *NotOp1; |
| int XIdx; |
| IntrinsicInst *II; |
| |
| if (!match(Op0, m_ICmp(Pred, m_Value(X), m_Zero()))) |
| return false; |
| |
| /// %Agg = call { i4, i1 } @llvm.[us]mul.with.overflow.i4(i4 %X, i4 %???) |
| /// %V = extractvalue { i4, i1 } %Agg, 1 |
| auto matchMulOverflowCheck = [X, &II, &XIdx](Value *V) { |
| auto *Extract = dyn_cast<ExtractValueInst>(V); |
| // We should only be extracting the overflow bit. |
| if (!Extract || !Extract->getIndices().equals(1)) |
| return false; |
| |
| II = dyn_cast<IntrinsicInst>(Extract->getAggregateOperand()); |
| if (!II || |
| !match(II, m_CombineOr(m_Intrinsic<Intrinsic::umul_with_overflow>(), |
| m_Intrinsic<Intrinsic::smul_with_overflow>()))) |
| return false; |
| |
| if (II->getArgOperand(0) == X) |
| XIdx = 0; |
| else if (II->getArgOperand(1) == X) |
| XIdx = 1; |
| else |
| return false; |
| return true; |
| }; |
| |
| bool Matched = |
| (IsAnd && Pred == ICmpInst::Predicate::ICMP_NE && |
| matchMulOverflowCheck(Op1)) || |
| (!IsAnd && Pred == ICmpInst::Predicate::ICMP_EQ && |
| match(Op1, m_Not(m_Value(NotOp1))) && matchMulOverflowCheck(NotOp1)); |
| |
| if (!Matched) |
| return false; |
| |
| Y = &II->getArgOperandUse(!XIdx); |
| return true; |
| } |
| |
| bool llvm::isCheckForZeroAndMulWithOverflow(Value *Op0, Value *Op1, |
| bool IsAnd) { |
| Use *Y; |
| return isCheckForZeroAndMulWithOverflow(Op0, Op1, IsAnd, Y); |
| } |