summaryrefslogtreecommitdiff
path: root/llvm/lib/Analysis/OverflowInstAnalysis.cpp
diff options
context:
space:
mode:
authorDimitry Andric <dim@FreeBSD.org>2021-07-29 20:15:26 +0000
committerDimitry Andric <dim@FreeBSD.org>2021-07-29 20:15:26 +0000
commit344a3780b2e33f6ca763666c380202b18aab72a3 (patch)
treef0b203ee6eb71d7fdd792373e3c81eb18d6934dd /llvm/lib/Analysis/OverflowInstAnalysis.cpp
parentb60736ec1405bb0a8dd40989f67ef4c93da068ab (diff)
Diffstat (limited to 'llvm/lib/Analysis/OverflowInstAnalysis.cpp')
-rw-r--r--llvm/lib/Analysis/OverflowInstAnalysis.cpp72
1 files changed, 72 insertions, 0 deletions
diff --git a/llvm/lib/Analysis/OverflowInstAnalysis.cpp b/llvm/lib/Analysis/OverflowInstAnalysis.cpp
new file mode 100644
index 000000000000..9f17d5b2064d
--- /dev/null
+++ b/llvm/lib/Analysis/OverflowInstAnalysis.cpp
@@ -0,0 +1,72 @@
+//==-- 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);
+} \ No newline at end of file