LLVM 22.0.0git
InstructionPrecedenceTracking.cpp
Go to the documentation of this file.
1//===-- InstructionPrecedenceTracking.cpp -----------------------*- 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// Implements a class that is able to define some instructions as "special"
9// (e.g. as having implicit control flow, or writing memory, or having another
10// interesting property) and then efficiently answers queries of the types:
11// 1. Are there any special instructions in the block of interest?
12// 2. Return first of the special instructions in the given block;
13// 3. Check if the given instruction is preceeded by the first special
14// instruction in the same block.
15// The class provides caching that allows to answer these queries quickly. The
16// user must make sure that the cached data is invalidated properly whenever
17// a content of some tracked block is changed.
18//===----------------------------------------------------------------------===//
19
24
25using namespace llvm;
26
27#ifndef NDEBUG
29 "ipt-expensive-asserts",
30 cl::desc("Perform expensive assert validation on every query to Instruction"
31 " Precedence Tracking"),
32 cl::init(false), cl::Hidden);
33#endif
34
36 const BasicBlock *BB) {
37#ifndef NDEBUG
38 // If there is a bug connected to invalid cache, turn on ExpensiveAsserts to
39 // catch this situation as early as possible.
41 validateAll();
42 else
43 validate(BB);
44#endif
45
46 auto [It, Inserted] = FirstSpecialInsts.try_emplace(BB);
47 if (Inserted) {
48 for (const auto &I : *BB) {
49 if (isSpecialInstruction(&I)) {
50 It->second = &I;
51 break;
52 }
53 }
54 }
55 return It->second;
56}
57
62
64 const Instruction *Insn) {
65 const Instruction *MaybeFirstSpecial =
67 return MaybeFirstSpecial && MaybeFirstSpecial->comesBefore(Insn);
68}
69
70#ifndef NDEBUG
71void InstructionPrecedenceTracking::validate(const BasicBlock *BB) const {
72 auto It = FirstSpecialInsts.find(BB);
73 // Bail if we don't have anything cached for this block.
74 if (It == FirstSpecialInsts.end())
75 return;
76
77 for (const Instruction &Insn : *BB)
78 if (isSpecialInstruction(&Insn)) {
79 assert(It->second == &Insn &&
80 "Cached first special instruction is wrong!");
81 return;
82 }
83
84 assert(It->second == nullptr &&
85 "Block is marked as having special instructions but in fact it has "
86 "none!");
87}
88
89void InstructionPrecedenceTracking::validateAll() const {
90 // Check that for every known block the cached value is correct.
91 for (const auto &It : FirstSpecialInsts)
92 validate(It.first);
93}
94#endif
95
97 const BasicBlock *BB) {
98 if (isSpecialInstruction(Inst))
99 FirstSpecialInsts.erase(BB);
100}
101
103 auto *BB = Inst->getParent();
104 assert(BB && "must be called before instruction is actually removed");
105 auto It = FirstSpecialInsts.find(BB);
106 if (It != FirstSpecialInsts.end() && It->second == Inst)
107 FirstSpecialInsts.erase(It);
108}
109
111 for (const auto *U : Inst->users()) {
112 if (const auto *UI = dyn_cast<Instruction>(U))
114 }
115}
116
118 FirstSpecialInsts.clear();
119#ifndef NDEBUG
120 // The map should be valid after clearing (at least empty).
121 validateAll();
122#endif
123}
124
126 const Instruction *Insn) const {
127 // If a block's instruction doesn't always pass the control to its successor
128 // instruction, mark the block as having implicit control flow. We use them
129 // to avoid wrong assumptions of sort "if A is executed and B post-dominates
130 // A, then B is also executed". This is not true is there is an implicit
131 // control flow instruction (e.g. a guard) between them.
133}
134
136 const Instruction *Insn) const {
137 using namespace PatternMatch;
139 return false;
140 return Insn->mayWriteToMemory();
141}
assert(UImm &&(UImm !=~static_cast< T >(0)) &&"Invalid immediate!")
static cl::opt< bool > ExpensiveAsserts("ipt-expensive-asserts", cl::desc("Perform expensive assert validation on every query to Instruction" " Precedence Tracking"), cl::init(false), cl::Hidden)
#define I(x, y, z)
Definition MD5.cpp:57
LLVM Basic Block Representation.
Definition BasicBlock.h:62
bool isSpecialInstruction(const Instruction *Insn) const override
A predicate that defines whether or not the instruction Insn is considered special and needs to be tr...
LLVM_ABI void clear()
Invalidates all information from this tracking.
LLVM_ABI void removeUsersOf(const Instruction *Inst)
Notifies this tracking that we are going to replace all uses of Inst.
LLVM_ABI void insertInstructionTo(const Instruction *Inst, const BasicBlock *BB)
Notifies this tracking that we are going to insert a new instruction Inst to the basic block BB.
LLVM_ABI void removeInstruction(const Instruction *Inst)
Notifies this tracking that we are going to remove the instruction Inst It makes all necessary update...
LLVM_ABI const Instruction * getFirstSpecialInstruction(const BasicBlock *BB)
Returns the topmost special instruction from the block BB.
virtual bool isSpecialInstruction(const Instruction *Insn) const =0
A predicate that defines whether or not the instruction Insn is considered special and needs to be tr...
LLVM_ABI bool hasSpecialInstructions(const BasicBlock *BB)
Returns true iff at least one instruction from the basic block BB is special.
LLVM_ABI bool isPreceededBySpecialInstruction(const Instruction *Insn)
Returns true iff the first special instruction of Insn's block exists and dominates Insn.
LLVM_ABI bool mayWriteToMemory() const LLVM_READONLY
Return true if this instruction may modify memory.
LLVM_ABI bool comesBefore(const Instruction *Other) const
Given an instruction Other in the same basic block as this instruction, return true if this instructi...
bool isSpecialInstruction(const Instruction *Insn) const override
A predicate that defines whether or not the instruction Insn is considered special and needs to be tr...
iterator_range< user_iterator > users()
Definition Value.h:426
const ParentTy * getParent() const
Definition ilist_node.h:34
bool match(Val *V, const Pattern &P)
IntrinsicID_match m_Intrinsic()
Match intrinsic calls like this: m_Intrinsic<Intrinsic::fabs>(m_Value(X))
initializer< Ty > init(const Ty &Val)
This is an optimization pass for GlobalISel generic memory operations.
decltype(auto) dyn_cast(const From &Val)
dyn_cast<X> - Return the argument parameter cast to the specified type.
Definition Casting.h:643
LLVM_ABI bool isGuaranteedToTransferExecutionToSuccessor(const Instruction *I)
Return true if this function can prove that the instruction I will always transfer execution to one o...