LLVM 17.0.0git
GuardUtils.cpp
Go to the documentation of this file.
1//===-- GuardUtils.cpp - Utils for work with guards -------------*- 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// Utils that are used to perform analyzes related to guards and their
9// conditions.
10//===----------------------------------------------------------------------===//
11
14
15using namespace llvm;
16using namespace llvm::PatternMatch;
17
18bool llvm::isGuard(const User *U) {
19 return match(U, m_Intrinsic<Intrinsic::experimental_guard>());
20}
21
23 Value *Condition, *WidenableCondition;
24 BasicBlock *GuardedBB, *DeoptBB;
25 return parseWidenableBranch(U, Condition, WidenableCondition, GuardedBB,
26 DeoptBB);
27}
28
30 Value *Condition, *WidenableCondition;
31 BasicBlock *GuardedBB, *DeoptBB;
32 if (!parseWidenableBranch(U, Condition, WidenableCondition, GuardedBB,
33 DeoptBB))
34 return false;
36 Visited.insert(DeoptBB);
37 do {
38 for (auto &Insn : *DeoptBB) {
39 if (match(&Insn, m_Intrinsic<Intrinsic::experimental_deoptimize>()))
40 return true;
41 if (Insn.mayHaveSideEffects())
42 return false;
43 }
44 DeoptBB = DeoptBB->getUniqueSuccessor();
45 if (!DeoptBB)
46 return false;
47 } while (Visited.insert(DeoptBB).second);
48 return false;
49}
50
51bool llvm::parseWidenableBranch(const User *U, Value *&Condition,
52 Value *&WidenableCondition,
53 BasicBlock *&IfTrueBB, BasicBlock *&IfFalseBB) {
54
55 Use *C, *WC;
56 if (parseWidenableBranch(const_cast<User*>(U), C, WC, IfTrueBB, IfFalseBB)) {
57 if (C)
58 Condition = C->get();
59 else
60 Condition = ConstantInt::getTrue(IfTrueBB->getContext());
61 WidenableCondition = WC->get();
62 return true;
63 }
64 return false;
65}
66
68 BasicBlock *&IfTrueBB, BasicBlock *&IfFalseBB) {
69
70 auto *BI = dyn_cast<BranchInst>(U);
71 if (!BI || !BI->isConditional())
72 return false;
73 auto *Cond = BI->getCondition();
74 if (!Cond->hasOneUse())
75 return false;
76
77 IfTrueBB = BI->getSuccessor(0);
78 IfFalseBB = BI->getSuccessor(1);
79
80 if (match(Cond, m_Intrinsic<Intrinsic::experimental_widenable_condition>())) {
81 WC = &BI->getOperandUse(0);
82 C = nullptr;
83 return true;
84 }
85
86 // Check for two cases:
87 // 1) br (i1 (and A, WC())), label %IfTrue, label %IfFalse
88 // 2) br (i1 (and WC(), B)), label %IfTrue, label %IfFalse
89 // We do not check for more generalized and trees as we should canonicalize
90 // to the form above in instcombine. (TODO)
91 Value *A, *B;
92 if (!match(Cond, m_And(m_Value(A), m_Value(B))))
93 return false;
94 auto *And = dyn_cast<Instruction>(Cond);
95 if (!And)
96 // Could be a constexpr
97 return false;
98
99 if (match(A, m_Intrinsic<Intrinsic::experimental_widenable_condition>()) &&
100 A->hasOneUse()) {
101 WC = &And->getOperandUse(0);
102 C = &And->getOperandUse(1);
103 return true;
104 }
105
106 if (match(B, m_Intrinsic<Intrinsic::experimental_widenable_condition>()) &&
107 B->hasOneUse()) {
108 WC = &And->getOperandUse(1);
109 C = &And->getOperandUse(0);
110 return true;
111 }
112 return false;
113}
SmallVector< AArch64_IMM::ImmInsnModel, 4 > Insn
SmallVector< MachineOperand, 4 > Cond
static GCRegistry::Add< OcamlGC > B("ocaml", "ocaml 3.10-compatible GC")
static GCRegistry::Add< ErlangGC > A("erlang", "erlang-compatible garbage collector")
LLVM Basic Block Representation.
Definition: BasicBlock.h:56
LLVMContext & getContext() const
Get the context in which this basic block lives.
Definition: BasicBlock.cpp:35
static ConstantInt * getTrue(LLVMContext &Context)
Definition: Constants.cpp:833
std::pair< iterator, bool > insert(PtrType Ptr)
Inserts Ptr if and only if there is no element in the container equal to Ptr.
Definition: SmallPtrSet.h:365
SmallPtrSet - This class implements a set which is optimized for holding SmallSize or less elements.
Definition: SmallPtrSet.h:450
A Use represents the edge between a Value definition and its users.
Definition: Use.h:43
LLVM Value Representation.
Definition: Value.h:74
@ C
The default llvm calling convention, compatible with C.
Definition: CallingConv.h:34
BinaryOp_match< LHS, RHS, Instruction::And > m_And(const LHS &L, const RHS &R)
bool match(Val *V, const Pattern &P)
Definition: PatternMatch.h:49
class_match< Value > m_Value()
Match an arbitrary value and ignore it.
Definition: PatternMatch.h:76
This is an optimization pass for GlobalISel generic memory operations.
Definition: AddressRanges.h:18
bool isGuard(const User *U)
Returns true iff U has semantics of a guard expressed in a form of call of llvm.experimental....
Definition: GuardUtils.cpp:18
bool parseWidenableBranch(const User *U, Value *&Condition, Value *&WidenableCondition, BasicBlock *&IfTrueBB, BasicBlock *&IfFalseBB)
If U is widenable branch looking like: cond = ... wc = call i1 @llvm.experimental....
Definition: GuardUtils.cpp:51
bool isWidenableBranch(const User *U)
Returns true iff U is a widenable branch (that is, parseWidenableBranch returns true).
Definition: GuardUtils.cpp:22
bool isGuardAsWidenableBranch(const User *U)
Returns true iff U has semantics of a guard expressed in a form of a widenable conditional branch to ...
Definition: GuardUtils.cpp:29
@ And
Bitwise or logical AND of integers.