Skip to content

Commit baea663

Browse files
committed
[IPT] Restructure cache to allow lazy update following invalidation [NFC]
This change restructures the cache used in IPT to point not to the first special instruction, but to the first instruction which *could* be special. That is, the cached reference is always equal to the first special, or comes before it in the block. This avoids expensive block scans when we are removing special instructions from the beginning of the block. At the moment, this case is not heavily used, though it does trigger in GVN when doing CSE of calls. The main motivation was a change I'm no longer planning to move forward with, but the cache optimization seemed worthwhile as a minor perf win at low cost. Differential Revision: https://reviews.llvm.org/D111768
1 parent acfe7d8 commit baea663

File tree

2 files changed

+40
-32
lines changed

2 files changed

+40
-32
lines changed

llvm/include/llvm/Analysis/InstructionPrecedenceTracking.h

+5-6
Original file line numberDiff line numberDiff line change
@@ -28,14 +28,13 @@ class BasicBlock;
2828
class Instruction;
2929

3030
class InstructionPrecedenceTracking {
31-
// Maps a block to the topmost special instruction in it. If the value is
32-
// nullptr, it means that it is known that this block does not contain any
33-
// special instructions.
31+
// Maps a block to the topmost instruction which *might* be a special
32+
// instruction in it. This value is lazily updated on query to point to the
33+
// topmost special instruction, but we allow it to point before that for
34+
// efficient invalidation. If the value is nullptr, it means that it is
35+
// known that this block does not contain any special instructions.
3436
DenseMap<const BasicBlock *, const Instruction *> FirstSpecialInsts;
3537

36-
// Fills information about the given block's special instructions.
37-
void fill(const BasicBlock *BB);
38-
3938
#ifndef NDEBUG
4039
/// Asserts that the cached info for \p BB is up-to-date. This helps to catch
4140
/// the usage error of accessing a block without properly invalidating after a

llvm/lib/Analysis/InstructionPrecedenceTracking.cpp

+35-26
Original file line numberDiff line numberDiff line change
@@ -47,11 +47,29 @@ const Instruction *InstructionPrecedenceTracking::getFirstSpecialInstruction(
4747
validate(BB);
4848
#endif
4949

50-
if (FirstSpecialInsts.find(BB) == FirstSpecialInsts.end()) {
51-
fill(BB);
52-
assert(FirstSpecialInsts.find(BB) != FirstSpecialInsts.end() && "Must be!");
53-
}
54-
return FirstSpecialInsts[BB];
50+
if (!FirstSpecialInsts.count(BB))
51+
// Seed the lazy scan
52+
FirstSpecialInsts[BB] = &*BB->begin();
53+
54+
auto *CurI = FirstSpecialInsts[BB];
55+
if (!CurI || isSpecialInstruction(CurI))
56+
// We found a cached definite result
57+
return CurI;
58+
59+
// Otherwise, scan forward until we find a definite result, then cache that.
60+
auto *Res = [&]() -> const Instruction * {
61+
for (auto &I : make_range(CurI->getIterator(), BB->end())) {
62+
NumInstScanned++;
63+
if (isSpecialInstruction(&I))
64+
// Found next special instruction
65+
return &I;
66+
}
67+
// Mark this block as having no special instructions.
68+
return nullptr;
69+
}();
70+
71+
FirstSpecialInsts[BB] = Res;
72+
return Res;
5573
}
5674

5775
bool InstructionPrecedenceTracking::hasSpecialInstructions(
@@ -66,33 +84,20 @@ bool InstructionPrecedenceTracking::isPreceededBySpecialInstruction(
6684
return MaybeFirstSpecial && MaybeFirstSpecial->comesBefore(Insn);
6785
}
6886

69-
void InstructionPrecedenceTracking::fill(const BasicBlock *BB) {
70-
FirstSpecialInsts.erase(BB);
71-
for (auto &I : *BB) {
72-
NumInstScanned++;
73-
if (isSpecialInstruction(&I)) {
74-
FirstSpecialInsts[BB] = &I;
75-
return;
76-
}
77-
}
78-
79-
// Mark this block as having no special instructions.
80-
FirstSpecialInsts[BB] = nullptr;
81-
}
82-
8387
#ifndef NDEBUG
8488
void InstructionPrecedenceTracking::validate(const BasicBlock *BB) const {
8589
auto It = FirstSpecialInsts.find(BB);
8690
// Bail if we don't have anything cached for this block.
8791
if (It == FirstSpecialInsts.end())
8892
return;
8993

90-
for (const Instruction &Insn : *BB)
91-
if (isSpecialInstruction(&Insn)) {
92-
assert(It->second == &Insn &&
93-
"Cached first special instruction is wrong!");
94+
for (const Instruction &I : *BB) {
95+
if (&I == It->second)
96+
// No special instruction before cached result
9497
return;
95-
}
98+
assert(!isSpecialInstruction(&I) &&
99+
"Cached first special instruction is wrong!");
100+
}
96101

97102
assert(It->second == nullptr &&
98103
"Block is marked as having special instructions but in fact it has "
@@ -115,8 +120,12 @@ void InstructionPrecedenceTracking::insertInstructionTo(const Instruction *Inst,
115120
void InstructionPrecedenceTracking::removeInstruction(const Instruction *Inst) {
116121
auto *BB = Inst->getParent();
117122
assert(BB && "must be called before instruction is actually removed");
118-
if (FirstSpecialInsts.count(BB) && FirstSpecialInsts[BB] == Inst)
119-
FirstSpecialInsts.erase(BB);
123+
if (FirstSpecialInsts.count(BB) && FirstSpecialInsts[BB] == Inst) {
124+
if (Inst->isTerminator())
125+
FirstSpecialInsts[BB] = nullptr;
126+
else
127+
FirstSpecialInsts[BB] = &*std::next(Inst->getIterator());
128+
}
120129
}
121130

122131
void InstructionPrecedenceTracking::removeUsersOf(const Instruction *Inst) {

0 commit comments

Comments
 (0)