LLVM 22.0.0git
Instruction.cpp
Go to the documentation of this file.
1//===--------------------- Instruction.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//
9// This file defines abstractions used by the Pipeline to model register reads,
10// register writes and instructions.
11//
12//===----------------------------------------------------------------------===//
13
15#include "llvm/Support/Debug.h"
17
18namespace llvm {
19namespace mca {
20
21void WriteState::writeStartEvent(unsigned IID, MCPhysReg RegID,
22 unsigned Cycles) {
23 CRD.IID = IID;
24 CRD.RegID = RegID;
25 CRD.Cycles = Cycles;
26 DependentWriteCyclesLeft = Cycles;
27 DependentWrite = nullptr;
28}
29
30void ReadState::writeStartEvent(unsigned IID, MCPhysReg RegID,
31 unsigned Cycles) {
32 assert(DependentWrites);
33 assert(CyclesLeft == UNKNOWN_CYCLES);
34
35 // This read may be dependent on more than one write. This typically occurs
36 // when a definition is the result of multiple writes where at least one
37 // write does a partial register update.
38 // The HW is forced to do some extra bookkeeping to track of all the
39 // dependent writes, and implement a merging scheme for the partial writes.
40 --DependentWrites;
41 if (TotalCycles < Cycles) {
42 CRD.IID = IID;
43 CRD.RegID = RegID;
44 CRD.Cycles = Cycles;
45 TotalCycles = Cycles;
46 }
47
48 if (!DependentWrites) {
49 CyclesLeft = TotalCycles;
50 IsReady = !CyclesLeft;
51 }
52}
53
55 assert(CyclesLeft == UNKNOWN_CYCLES);
56 // Update the number of cycles left based on the WriteDescriptor info.
57 CyclesLeft = getLatency();
58
59 // Now that the time left before write-back is known, notify
60 // all the users.
61 for (const std::pair<ReadState *, int> &User : Users) {
62 ReadState *RS = User.first;
63 unsigned ReadCycles = std::max(0, CyclesLeft - User.second);
64 RS->writeStartEvent(IID, RegisterID, ReadCycles);
65 }
66
67 // Notify any writes that are in a false dependency with this write.
68 if (PartialWrite)
69 PartialWrite->writeStartEvent(IID, RegisterID, CyclesLeft);
70}
71
72void WriteState::addUser(unsigned IID, ReadState *User, int ReadAdvance) {
73 // If CyclesLeft is different than -1, then we don't need to
74 // update the list of users. We can just notify the user with
75 // the actual number of cycles left (which may be zero).
76 if (CyclesLeft != UNKNOWN_CYCLES) {
77 unsigned ReadCycles = std::max(0, CyclesLeft - ReadAdvance);
78 User->writeStartEvent(IID, RegisterID, ReadCycles);
79 return;
80 }
81
82 Users.emplace_back(User, ReadAdvance);
83}
84
85void WriteState::addUser(unsigned IID, WriteState *User) {
86 if (CyclesLeft != UNKNOWN_CYCLES) {
87 User->writeStartEvent(IID, RegisterID, std::max(0, CyclesLeft));
88 return;
89 }
90
91 assert(!PartialWrite && "PartialWrite already set!");
92 PartialWrite = User;
93 User->setDependentWrite(this);
94}
95
97 // Note: CyclesLeft can be a negative number. It is an error to
98 // make it an unsigned quantity because users of this write may
99 // specify a negative ReadAdvance.
100 if (CyclesLeft != UNKNOWN_CYCLES)
101 CyclesLeft--;
102
103 if (DependentWriteCyclesLeft)
104 DependentWriteCyclesLeft--;
105}
106
108 // Update the total number of cycles.
109 if (DependentWrites && TotalCycles) {
110 --TotalCycles;
111 return;
112 }
113
114 // Bail out immediately if we don't know how many cycles are left.
115 if (CyclesLeft == UNKNOWN_CYCLES)
116 return;
117
118 if (CyclesLeft) {
119 --CyclesLeft;
120 IsReady = !CyclesLeft;
121 }
122}
123
124#ifndef NDEBUG
125void WriteState::dump() const {
126 dbgs() << "{ OpIdx=" << WD->OpIndex << ", Lat=" << getLatency() << ", RegID "
127 << getRegisterID() << ", Cycles Left=" << getCyclesLeft() << " }";
128}
129#endif
130
131#ifndef NDEBUG
132void ReadState::dump() const {
133 dbgs() << "{ OpIdx=" << RD->OpIndex << ", RegID " << getRegisterID()
134 << ", Cycles Left=" << CyclesLeft << " }";
135}
136#endif
137
139 if (CriticalRegDep.Cycles)
140 return CriticalRegDep;
141
142 unsigned MaxLatency = 0;
143 for (const WriteState &WS : getDefs()) {
144 const CriticalDependency &WriteCRD = WS.getCriticalRegDep();
145 if (WriteCRD.Cycles > MaxLatency)
146 CriticalRegDep = WriteCRD;
147 }
148
149 for (const ReadState &RS : getUses()) {
150 const CriticalDependency &ReadCRD = RS.getCriticalRegDep();
151 if (ReadCRD.Cycles > MaxLatency)
152 CriticalRegDep = ReadCRD;
153 }
154
155 return CriticalRegDep;
156}
157
159 // Note that this won't clear read/write descriptors
160 // or other non-trivial fields
161 Stage = IS_INVALID;
162 CyclesLeft = UNKNOWN_CYCLES;
164 RCUTokenID = 0;
165 LSUTokenID = 0;
166 CriticalResourceMask = 0;
167 IsEliminated = false;
168}
169
170void Instruction::dispatch(unsigned RCUToken) {
171 assert(Stage == IS_INVALID);
172 Stage = IS_DISPATCHED;
173 RCUTokenID = RCUToken;
174
175 // Check if input operands are already available.
176 if (updateDispatched())
178}
179
180void Instruction::execute(unsigned IID) {
181 assert(Stage == IS_READY);
182 Stage = IS_EXECUTING;
183
184 // Set the cycles left before the write-back stage.
185 CyclesLeft = getLatency();
186
187 for (WriteState &WS : getDefs())
188 WS.onInstructionIssued(IID);
189
190 // Transition to the "executed" stage if this is a zero-latency instruction.
191 if (!CyclesLeft)
192 Stage = IS_EXECUTED;
193}
194
196 assert(Stage == IS_READY && "Invalid internal state!");
197 CyclesLeft = 0;
198 Stage = IS_EXECUTED;
199}
200
202 assert(isPending() && "Unexpected instruction stage found!");
203
204 if (!all_of(getUses(), [](const ReadState &Use) { return Use.isReady(); }))
205 return false;
206
207 // A partial register write cannot complete before a dependent write.
208 if (!all_of(getDefs(), [](const WriteState &Def) { return Def.isReady(); }))
209 return false;
210
211 Stage = IS_READY;
212 return true;
213}
214
216 assert(isDispatched() && "Unexpected instruction stage found!");
217
218 if (!all_of(getUses(), [](const ReadState &Use) {
219 return Use.isPending() || Use.isReady();
220 }))
221 return false;
222
223 // A partial register write cannot complete before a dependent write.
224 if (!all_of(getDefs(),
225 [](const WriteState &Def) { return !Def.getDependentWrite(); }))
226 return false;
227
228 Stage = IS_PENDING;
229 return true;
230}
231
233 if (isDispatched())
235 if (isPending())
237}
238
240 if (isReady())
241 return;
242
243 if (isDispatched() || isPending()) {
244 for (ReadState &Use : getUses())
245 Use.cycleEvent();
246
247 for (WriteState &Def : getDefs())
248 Def.cycleEvent();
249
250 update();
251 return;
252 }
253
254 assert(isExecuting() && "Instruction not in-flight?");
255 assert(CyclesLeft && "Instruction already executed?");
256 for (WriteState &Def : getDefs())
257 Def.cycleEvent();
258 CyclesLeft--;
259 if (!CyclesLeft)
260 Stage = IS_EXECUTED;
261}
262
263} // namespace mca
264} // namespace llvm
assert(UImm &&(UImm !=~static_cast< T >(0)) &&"Invalid immediate!")
This file defines abstractions used by the Pipeline to model register reads, register writes and inst...
A Use represents the edge between a Value definition and its users.
Definition: Use.h:35
SmallVectorImpl< WriteState > & getDefs()
Definition: Instruction.h:540
unsigned getLatency() const
Definition: Instruction.h:546
SmallVectorImpl< ReadState > & getUses()
Definition: Instruction.h:542
LLVM_ABI void cycleEvent()
bool isDispatched() const
Definition: Instruction.h:690
LLVM_ABI bool updateDispatched()
bool isExecuting() const
Definition: Instruction.h:693
LLVM_ABI void forceExecuted()
bool isReady() const
Definition: Instruction.h:692
LLVM_ABI const CriticalDependency & computeCriticalRegDep()
LLVM_ABI void execute(unsigned IID)
LLVM_ABI bool updatePending()
LLVM_ABI void dispatch(unsigned RCUTokenID)
LLVM_ABI void update()
LLVM_ABI void reset()
bool isPending() const
Definition: Instruction.h:691
Tracks register operand latency in cycles.
Definition: Instruction.h:327
LLVM_ABI void writeStartEvent(unsigned IID, MCPhysReg RegID, unsigned Cycles)
Definition: Instruction.cpp:30
LLVM_ABI void cycleEvent()
MCPhysReg getRegisterID() const
Definition: Instruction.h:364
Tracks uses of a register definition (e.g.
Definition: Instruction.h:198
LLVM_ABI void writeStartEvent(unsigned IID, MCPhysReg RegID, unsigned Cycles)
Definition: Instruction.cpp:21
unsigned getLatency() const
Definition: Instruction.h:264
int getCyclesLeft() const
Definition: Instruction.h:259
LLVM_ABI void cycleEvent()
Definition: Instruction.cpp:96
LLVM_ABI void onInstructionIssued(unsigned IID)
Definition: Instruction.cpp:54
MCPhysReg getRegisterID() const
Definition: Instruction.h:261
LLVM_ABI void addUser(unsigned IID, ReadState *Use, int ReadAdvance)
Definition: Instruction.cpp:72
constexpr int UNKNOWN_CYCLES
Definition: Instruction.h:35
This is an optimization pass for GlobalISel generic memory operations.
Definition: AddressRanges.h:18
bool all_of(R &&range, UnaryPredicate P)
Provide wrappers to std::all_of which take ranges instead of having to pass begin/end explicitly.
Definition: STLExtras.h:1744
LLVM_ABI raw_ostream & dbgs()
dbgs() - This returns a reference to a raw_ostream for debugging messages.
Definition: Debug.cpp:207
A critical data dependency descriptor.
Definition: Instruction.h:186