1 //===----------------------- LSUnit.cpp --------------------------*- C++-*-===//
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
7 //===----------------------------------------------------------------------===//
10 /// A Load-Store Unit for the llvm-mca tool.
12 //===----------------------------------------------------------------------===//
14 #include "llvm/MCA/HardwareUnits/LSUnit.h"
15 #include "llvm/MCA/Instruction.h"
16 #include "llvm/Support/Debug.h"
17 #include "llvm/Support/raw_ostream.h"
19 #define DEBUG_TYPE "llvm-mca"
24 LSUnitBase::LSUnitBase(const MCSchedModel
&SM
, unsigned LQ
, unsigned SQ
,
26 : LQSize(LQ
), SQSize(SQ
), UsedLQEntries(0), UsedSQEntries(0),
27 NoAlias(AssumeNoAlias
), NextGroupID(1) {
28 if (SM
.hasExtraProcessorInfo()) {
29 const MCExtraProcessorInfo
&EPI
= SM
.getExtraProcessorInfo();
30 if (!LQSize
&& EPI
.LoadQueueID
) {
31 const MCProcResourceDesc
&LdQDesc
= *SM
.getProcResource(EPI
.LoadQueueID
);
32 LQSize
= std::max(0, LdQDesc
.BufferSize
);
35 if (!SQSize
&& EPI
.StoreQueueID
) {
36 const MCProcResourceDesc
&StQDesc
= *SM
.getProcResource(EPI
.StoreQueueID
);
37 SQSize
= std::max(0, StQDesc
.BufferSize
);
42 LSUnitBase::~LSUnitBase() = default;
44 void LSUnitBase::cycleEvent() {
45 for (const std::pair
<unsigned, std::unique_ptr
<MemoryGroup
>> &G
: Groups
)
46 G
.second
->cycleEvent();
50 void LSUnitBase::dump() const {
51 dbgs() << "[LSUnit] LQ_Size = " << getLoadQueueSize() << '\n';
52 dbgs() << "[LSUnit] SQ_Size = " << getStoreQueueSize() << '\n';
53 dbgs() << "[LSUnit] NextLQSlotIdx = " << getUsedLQEntries() << '\n';
54 dbgs() << "[LSUnit] NextSQSlotIdx = " << getUsedSQEntries() << '\n';
56 for (const auto &GroupIt
: Groups
) {
57 const MemoryGroup
&Group
= *GroupIt
.second
;
58 dbgs() << "[LSUnit] Group (" << GroupIt
.first
<< "): "
59 << "[ #Preds = " << Group
.getNumPredecessors()
60 << ", #GIssued = " << Group
.getNumExecutingPredecessors()
61 << ", #GExecuted = " << Group
.getNumExecutedPredecessors()
62 << ", #Inst = " << Group
.getNumInstructions()
63 << ", #IIssued = " << Group
.getNumExecuting()
64 << ", #IExecuted = " << Group
.getNumExecuted() << '\n';
69 unsigned LSUnit::dispatch(const InstRef
&IR
) {
70 const Instruction
&IS
= *IR
.getInstruction();
71 bool IsStoreBarrier
= IS
.isAStoreBarrier();
72 bool IsLoadBarrier
= IS
.isALoadBarrier();
73 assert((IS
.getMayLoad() || IS
.getMayStore()) && "Not a memory operation!");
80 if (IS
.getMayStore()) {
81 unsigned NewGID
= createMemoryGroup();
82 MemoryGroup
&NewGroup
= getGroup(NewGID
);
83 NewGroup
.addInstruction();
85 // A store may not pass a previous load or load barrier.
86 unsigned ImmediateLoadDominator
=
87 std::max(CurrentLoadGroupID
, CurrentLoadBarrierGroupID
);
88 if (ImmediateLoadDominator
) {
89 MemoryGroup
&IDom
= getGroup(ImmediateLoadDominator
);
90 LLVM_DEBUG(dbgs() << "[LSUnit]: GROUP DEP: (" << ImmediateLoadDominator
91 << ") --> (" << NewGID
<< ")\n");
92 IDom
.addSuccessor(&NewGroup
, !assumeNoAlias());
95 // A store may not pass a previous store barrier.
96 if (CurrentStoreBarrierGroupID
) {
97 MemoryGroup
&StoreGroup
= getGroup(CurrentStoreBarrierGroupID
);
98 LLVM_DEBUG(dbgs() << "[LSUnit]: GROUP DEP: ("
99 << CurrentStoreBarrierGroupID
100 << ") --> (" << NewGID
<< ")\n");
101 StoreGroup
.addSuccessor(&NewGroup
, true);
104 // A store may not pass a previous store.
105 if (CurrentStoreGroupID
&&
106 (CurrentStoreGroupID
!= CurrentStoreBarrierGroupID
)) {
107 MemoryGroup
&StoreGroup
= getGroup(CurrentStoreGroupID
);
108 LLVM_DEBUG(dbgs() << "[LSUnit]: GROUP DEP: (" << CurrentStoreGroupID
109 << ") --> (" << NewGID
<< ")\n");
110 StoreGroup
.addSuccessor(&NewGroup
, !assumeNoAlias());
114 CurrentStoreGroupID
= NewGID
;
116 CurrentStoreBarrierGroupID
= NewGID
;
118 if (IS
.getMayLoad()) {
119 CurrentLoadGroupID
= NewGID
;
121 CurrentLoadBarrierGroupID
= NewGID
;
127 assert(IS
.getMayLoad() && "Expected a load!");
129 unsigned ImmediateLoadDominator
=
130 std::max(CurrentLoadGroupID
, CurrentLoadBarrierGroupID
);
132 // A new load group is created if we are in one of the following situations:
133 // 1) This is a load barrier (by construction, a load barrier is always
134 // assigned to a different memory group).
135 // 2) There is no load in flight (by construction we always keep loads and
136 // stores into separate memory groups).
137 // 3) There is a load barrier in flight. This load depends on it.
138 // 4) There is an intervening store between the last load dispatched to the
139 // LSU and this load. We always create a new group even if this load
140 // does not alias the last dispatched store.
141 // 5) There is no intervening store and there is an active load group.
142 // However that group has already started execution, so we cannot add
144 bool ShouldCreateANewGroup
=
145 IsLoadBarrier
|| !ImmediateLoadDominator
||
146 CurrentLoadBarrierGroupID
== ImmediateLoadDominator
||
147 ImmediateLoadDominator
<= CurrentStoreGroupID
||
148 getGroup(ImmediateLoadDominator
).isExecuting();
150 if (ShouldCreateANewGroup
) {
151 unsigned NewGID
= createMemoryGroup();
152 MemoryGroup
&NewGroup
= getGroup(NewGID
);
153 NewGroup
.addInstruction();
155 // A load may not pass a previous store or store barrier
156 // unless flag 'NoAlias' is set.
157 if (!assumeNoAlias() && CurrentStoreGroupID
) {
158 MemoryGroup
&StoreGroup
= getGroup(CurrentStoreGroupID
);
159 LLVM_DEBUG(dbgs() << "[LSUnit]: GROUP DEP: (" << CurrentStoreGroupID
160 << ") --> (" << NewGID
<< ")\n");
161 StoreGroup
.addSuccessor(&NewGroup
, true);
164 // A load barrier may not pass a previous load or load barrier.
166 if (ImmediateLoadDominator
) {
167 MemoryGroup
&LoadGroup
= getGroup(ImmediateLoadDominator
);
168 LLVM_DEBUG(dbgs() << "[LSUnit]: GROUP DEP: ("
169 << ImmediateLoadDominator
170 << ") --> (" << NewGID
<< ")\n");
171 LoadGroup
.addSuccessor(&NewGroup
, true);
174 // A younger load cannot pass a older load barrier.
175 if (CurrentLoadBarrierGroupID
) {
176 MemoryGroup
&LoadGroup
= getGroup(CurrentLoadBarrierGroupID
);
177 LLVM_DEBUG(dbgs() << "[LSUnit]: GROUP DEP: ("
178 << CurrentLoadBarrierGroupID
179 << ") --> (" << NewGID
<< ")\n");
180 LoadGroup
.addSuccessor(&NewGroup
, true);
184 CurrentLoadGroupID
= NewGID
;
186 CurrentLoadBarrierGroupID
= NewGID
;
190 // A load may pass a previous load.
191 MemoryGroup
&Group
= getGroup(CurrentLoadGroupID
);
192 Group
.addInstruction();
193 return CurrentLoadGroupID
;
196 LSUnit::Status
LSUnit::isAvailable(const InstRef
&IR
) const {
197 const Instruction
&IS
= *IR
.getInstruction();
198 if (IS
.getMayLoad() && isLQFull())
199 return LSUnit::LSU_LQUEUE_FULL
;
200 if (IS
.getMayStore() && isSQFull())
201 return LSUnit::LSU_SQUEUE_FULL
;
202 return LSUnit::LSU_AVAILABLE
;
205 void LSUnitBase::onInstructionExecuted(const InstRef
&IR
) {
206 unsigned GroupID
= IR
.getInstruction()->getLSUTokenID();
207 auto It
= Groups
.find(GroupID
);
208 assert(It
!= Groups
.end() && "Instruction not dispatched to the LS unit");
209 It
->second
->onInstructionExecuted(IR
);
210 if (It
->second
->isExecuted())
214 void LSUnitBase::onInstructionRetired(const InstRef
&IR
) {
215 const Instruction
&IS
= *IR
.getInstruction();
216 bool IsALoad
= IS
.getMayLoad();
217 bool IsAStore
= IS
.getMayStore();
218 assert((IsALoad
|| IsAStore
) && "Expected a memory operation!");
222 LLVM_DEBUG(dbgs() << "[LSUnit]: Instruction idx=" << IR
.getSourceIndex()
223 << " has been removed from the load queue.\n");
228 LLVM_DEBUG(dbgs() << "[LSUnit]: Instruction idx=" << IR
.getSourceIndex()
229 << " has been removed from the store queue.\n");
233 void LSUnit::onInstructionExecuted(const InstRef
&IR
) {
234 const Instruction
&IS
= *IR
.getInstruction();
238 LSUnitBase::onInstructionExecuted(IR
);
239 unsigned GroupID
= IS
.getLSUTokenID();
240 if (!isValidGroupID(GroupID
)) {
241 if (GroupID
== CurrentLoadGroupID
)
242 CurrentLoadGroupID
= 0;
243 if (GroupID
== CurrentStoreGroupID
)
244 CurrentStoreGroupID
= 0;
245 if (GroupID
== CurrentLoadBarrierGroupID
)
246 CurrentLoadBarrierGroupID
= 0;
247 if (GroupID
== CurrentStoreBarrierGroupID
)
248 CurrentStoreBarrierGroupID
= 0;