[ORC] Add std::tuple support to SimplePackedSerialization.
[llvm-project.git] / llvm / lib / Analysis / OverflowInstAnalysis.cpp
blob9f17d5b2064db652a7513ff0909a97e91ca2e465
1 //==-- OverflowInstAnalysis.cpp - Utils to fold overflow insts ----*- C++ -*-=//
2 //
3 // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
4 // See https://llvm.org/LICENSE.txt for license information.
5 // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
6 //
7 //===----------------------------------------------------------------------===//
8 //
9 // This file holds routines to help analyse overflow instructions
10 // and fold them into constants or other overflow instructions
12 //===----------------------------------------------------------------------===//
14 #include "llvm/Analysis/OverflowInstAnalysis.h"
15 #include "llvm/IR/Constants.h"
16 #include "llvm/IR/Instructions.h"
17 #include "llvm/IR/PatternMatch.h"
19 using namespace llvm;
20 using namespace llvm::PatternMatch;
22 bool llvm::isCheckForZeroAndMulWithOverflow(Value *Op0, Value *Op1, bool IsAnd,
23 Use *&Y) {
24 ICmpInst::Predicate Pred;
25 Value *X, *NotOp1;
26 int XIdx;
27 IntrinsicInst *II;
29 if (!match(Op0, m_ICmp(Pred, m_Value(X), m_Zero())))
30 return false;
32 /// %Agg = call { i4, i1 } @llvm.[us]mul.with.overflow.i4(i4 %X, i4 %???)
33 /// %V = extractvalue { i4, i1 } %Agg, 1
34 auto matchMulOverflowCheck = [X, &II, &XIdx](Value *V) {
35 auto *Extract = dyn_cast<ExtractValueInst>(V);
36 // We should only be extracting the overflow bit.
37 if (!Extract || !Extract->getIndices().equals(1))
38 return false;
40 II = dyn_cast<IntrinsicInst>(Extract->getAggregateOperand());
41 if (!II ||
42 !match(II, m_CombineOr(m_Intrinsic<Intrinsic::umul_with_overflow>(),
43 m_Intrinsic<Intrinsic::smul_with_overflow>())))
44 return false;
46 if (II->getArgOperand(0) == X)
47 XIdx = 0;
48 else if (II->getArgOperand(1) == X)
49 XIdx = 1;
50 else
51 return false;
52 return true;
55 bool Matched =
56 (IsAnd && Pred == ICmpInst::Predicate::ICMP_NE &&
57 matchMulOverflowCheck(Op1)) ||
58 (!IsAnd && Pred == ICmpInst::Predicate::ICMP_EQ &&
59 match(Op1, m_Not(m_Value(NotOp1))) && matchMulOverflowCheck(NotOp1));
61 if (!Matched)
62 return false;
64 Y = &II->getArgOperandUse(!XIdx);
65 return true;
68 bool llvm::isCheckForZeroAndMulWithOverflow(Value *Op0, Value *Op1,
69 bool IsAnd) {
70 Use *Y;
71 return isCheckForZeroAndMulWithOverflow(Op0, Op1, IsAnd, Y);