41 #ifndef __CPU_O3_ROB_IMPL_HH__
42 #define __CPU_O3_ROB_IMPL_HH__
48 #include "debug/Fetch.hh"
49 #include "debug/ROB.hh"
50 #include "params/DerivO3CPU.hh"
54 : robPolicy(params.smtROBPolicy),
56 numEntries(params.numROBEntries),
57 squashWidth(params.squashWidth),
59 numThreads(params.numThreads),
63 if (
robPolicy == SMTQueuePolicy::Dynamic) {
69 }
else if (
robPolicy == SMTQueuePolicy::Partitioned) {
70 DPRINTF(Fetch,
"ROB sharing policy set to Partitioned\n");
80 }
else if (
robPolicy == SMTQueuePolicy::Threshold) {
81 DPRINTF(Fetch,
"ROB sharing policy set to Threshold\n");
83 int threshold = params.smtROBThreshold;;
102 for (
ThreadID tid = 0; tid < Impl::MaxThreads; tid++) {
103 threadEntries[tid] = 0;
104 squashIt[tid] = instList[tid].end();
105 squashedSeqNum[tid] = 0;
106 doneSquashing[tid] =
true;
112 head = instList[0].end();
113 tail = instList[0].end();
116 template <
class Impl>
120 return cpu->name() +
".rob";
123 template <
class Impl>
127 DPRINTF(
ROB,
"Setting active threads list pointer.\n");
128 activeThreads = at_ptr;
131 template <
class Impl>
135 for (
ThreadID tid = 0; tid < numThreads; tid++)
136 assert(instList[tid].empty());
140 template <
class Impl>
147 template <
class Impl>
151 if (robPolicy != SMTQueuePolicy::Dynamic || numThreads > 1) {
152 auto active_threads = activeThreads->size();
157 while (threads != end) {
160 if (robPolicy == SMTQueuePolicy::Partitioned) {
161 maxEntries[tid] = numEntries / active_threads;
162 }
else if (robPolicy == SMTQueuePolicy::Threshold &&
163 active_threads == 1) {
164 maxEntries[tid] = numEntries;
170 template <
class Impl>
174 if (robPolicy == SMTQueuePolicy::Partitioned) {
175 return numEntries / num_threads;
181 template <
class Impl>
187 for (
ThreadID tid = 0; tid < numThreads; tid++)
188 total += countInsts(tid);
193 template <
class Impl>
197 return instList[tid].size();
200 template <
class Impl>
208 DPRINTF(
ROB,
"Adding inst PC %s to the ROB.\n", inst->pcState());
210 assert(numInstsInROB != numEntries);
214 instList[tid].push_back(inst);
217 if (numInstsInROB == 0) {
218 head = instList[tid].begin();
219 assert((*head) == inst);
224 tail = instList[tid].end();
230 ++threadEntries[tid];
232 assert((*tail) == inst);
234 DPRINTF(
ROB,
"[tid:%i] Now has %d instructions.\n", tid, threadEntries[tid]);
237 template <
class Impl>
243 assert(numInstsInROB > 0);
246 InstIt head_it = instList[tid].begin();
249 instList[tid].erase(head_it);
251 assert(head_inst->readyToCommit());
253 DPRINTF(
ROB,
"[tid:%i] Retiring head instruction, "
254 "instruction PC %s, [sn:%llu]\n", tid, head_inst->pcState(),
258 --threadEntries[tid];
260 head_inst->clearInROB();
261 head_inst->setCommitted();
269 cpu->removeFrontInst(head_inst);
272 template <
class Impl>
277 if (threadEntries[tid] != 0) {
278 return instList[tid].front()->readyToCommit();
284 template <
class Impl>
292 while (threads != end) {
295 if (isHeadReady(tid)) {
303 template <
class Impl>
307 return numEntries - numInstsInROB;
310 template <
class Impl>
314 return maxEntries[tid] - threadEntries[tid];
317 template <
class Impl>
322 DPRINTF(
ROB,
"[tid:%i] Squashing instructions until [sn:%llu].\n",
323 tid, squashedSeqNum[tid]);
325 assert(squashIt[tid] != instList[tid].end());
327 if ((*squashIt[tid])->seqNum < squashedSeqNum[tid]) {
328 DPRINTF(
ROB,
"[tid:%i] Done squashing instructions.\n",
331 squashIt[tid] = instList[tid].end();
333 doneSquashing[tid] =
true;
337 bool robTailUpdate =
false;
339 unsigned int numInstsToSquash = squashWidth;
344 if (cpu->isThreadExiting(tid))
346 numInstsToSquash = numEntries;
349 for (
int numSquashed = 0;
350 numSquashed < numInstsToSquash &&
351 squashIt[tid] != instList[tid].end() &&
352 (*squashIt[tid])->seqNum > squashedSeqNum[tid];
355 DPRINTF(
ROB,
"[tid:%i] Squashing instruction PC %s, seq num %i.\n",
356 (*squashIt[tid])->threadNumber,
357 (*squashIt[tid])->pcState(),
358 (*squashIt[tid])->seqNum);
362 (*squashIt[tid])->setSquashed();
364 (*squashIt[tid])->setCanCommit();
367 if (squashIt[tid] == instList[tid].begin()) {
368 DPRINTF(
ROB,
"Reached head of instruction list while "
371 squashIt[tid] = instList[tid].end();
373 doneSquashing[tid] =
true;
378 InstIt tail_thread = instList[tid].end();
381 if ((*squashIt[tid]) == (*tail_thread))
382 robTailUpdate =
true;
389 if ((*squashIt[tid])->seqNum <= squashedSeqNum[tid]) {
390 DPRINTF(
ROB,
"[tid:%i] Done squashing instructions.\n",
393 squashIt[tid] = instList[tid].end();
395 doneSquashing[tid] =
true;
404 template <
class Impl>
409 bool first_valid =
true;
415 while (threads != end) {
418 if (instList[tid].empty())
422 head = instList[tid].begin();
423 lowest_num = (*head)->seqNum;
428 InstIt head_thread = instList[tid].begin();
432 assert(head_inst != 0);
434 if (head_inst->seqNum < lowest_num) {
436 lowest_num = head_inst->seqNum;
441 head = instList[0].end();
446 template <
class Impl>
450 tail = instList[0].end();
451 bool first_valid =
true;
456 while (threads != end) {
459 if (instList[tid].empty()) {
466 tail = instList[tid].end();
474 InstIt tail_thread = instList[tid].end();
477 if ((*tail_thread)->seqNum > (*tail)->seqNum) {
484 template <
class Impl>
489 DPRINTF(
ROB,
"Does not need to squash due to being empty "
496 DPRINTF(
ROB,
"Starting to squash within the ROB.\n");
498 robStatus[tid] = ROBSquashing;
500 doneSquashing[tid] =
false;
502 squashedSeqNum[tid] = squash_num;
504 if (!instList[tid].empty()) {
505 InstIt tail_thread = instList[tid].end();
508 squashIt[tid] = tail_thread;
514 template <
class Impl>
515 const typename Impl::DynInstPtr&
518 if (threadEntries[tid] != 0) {
519 InstIt head_thread = instList[tid].begin();
521 assert((*head_thread)->isInROB());
529 template <
class Impl>
530 typename Impl::DynInstPtr
533 InstIt tail_thread = instList[tid].end();
539 template <
class Impl>
541 :
Stats::Group(parent,
"rob"),
547 template <
class Impl>
548 typename Impl::DynInstPtr
552 if ((*it)->seqNum == squash_inst) {
559 #endif//__CPU_O3_ROB_IMPL_HH__