gem5  v21.0.1.0
All Classes Namespaces Files Functions Variables Typedefs Enumerations Enumerator Friends Macros Modules Pages
rob_impl.hh
Go to the documentation of this file.
1 /*
2  * Copyright (c) 2012 ARM Limited
3  * All rights reserved
4  *
5  * The license below extends only to copyright in the software and shall
6  * not be construed as granting a license to any other intellectual
7  * property including but not limited to intellectual property relating
8  * to a hardware implementation of the functionality of the software
9  * licensed hereunder. You may use the software subject to the license
10  * terms below provided that you ensure that this notice is replicated
11  * unmodified and in its entirety in all distributions of the software,
12  * modified or unmodified, in source code or in binary form.
13  *
14  * Copyright (c) 2004-2006 The Regents of The University of Michigan
15  * All rights reserved.
16  *
17  * Redistribution and use in source and binary forms, with or without
18  * modification, are permitted provided that the following conditions are
19  * met: redistributions of source code must retain the above copyright
20  * notice, this list of conditions and the following disclaimer;
21  * redistributions in binary form must reproduce the above copyright
22  * notice, this list of conditions and the following disclaimer in the
23  * documentation and/or other materials provided with the distribution;
24  * neither the name of the copyright holders nor the names of its
25  * contributors may be used to endorse or promote products derived from
26  * this software without specific prior written permission.
27  *
28  * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
29  * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
30  * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
31  * A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
32  * OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
33  * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
34  * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
35  * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
36  * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
37  * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
38  * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
39  */
40 
41 #ifndef __CPU_O3_ROB_IMPL_HH__
42 #define __CPU_O3_ROB_IMPL_HH__
43 
44 #include <list>
45 
46 #include "base/logging.hh"
47 #include "cpu/o3/rob.hh"
48 #include "debug/Fetch.hh"
49 #include "debug/ROB.hh"
50 #include "params/DerivO3CPU.hh"
51 
52 template <class Impl>
53 ROB<Impl>::ROB(O3CPU *_cpu, const DerivO3CPUParams &params)
54  : robPolicy(params.smtROBPolicy),
55  cpu(_cpu),
56  numEntries(params.numROBEntries),
57  squashWidth(params.squashWidth),
58  numInstsInROB(0),
59  numThreads(params.numThreads),
60  stats(_cpu)
61 {
62  //Figure out rob policy
63  if (robPolicy == SMTQueuePolicy::Dynamic) {
64  //Set Max Entries to Total ROB Capacity
65  for (ThreadID tid = 0; tid < numThreads; tid++) {
66  maxEntries[tid] = numEntries;
67  }
68 
69  } else if (robPolicy == SMTQueuePolicy::Partitioned) {
70  DPRINTF(Fetch, "ROB sharing policy set to Partitioned\n");
71 
72  //@todo:make work if part_amt doesnt divide evenly.
73  int part_amt = numEntries / numThreads;
74 
75  //Divide ROB up evenly
76  for (ThreadID tid = 0; tid < numThreads; tid++) {
77  maxEntries[tid] = part_amt;
78  }
79 
80  } else if (robPolicy == SMTQueuePolicy::Threshold) {
81  DPRINTF(Fetch, "ROB sharing policy set to Threshold\n");
82 
83  int threshold = params.smtROBThreshold;;
84 
85  //Divide up by threshold amount
86  for (ThreadID tid = 0; tid < numThreads; tid++) {
87  maxEntries[tid] = threshold;
88  }
89  }
90 
91  for (ThreadID tid = numThreads; tid < Impl::MaxThreads; tid++) {
92  maxEntries[tid] = 0;
93  }
94 
95  resetState();
96 }
97 
98 template <class Impl>
99 void
101 {
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;
107  }
108  numInstsInROB = 0;
109 
110  // Initialize the "universal" ROB head & tail point to invalid
111  // pointers
112  head = instList[0].end();
113  tail = instList[0].end();
114 }
115 
116 template <class Impl>
117 std::string
119 {
120  return cpu->name() + ".rob";
121 }
122 
123 template <class Impl>
124 void
126 {
127  DPRINTF(ROB, "Setting active threads list pointer.\n");
128  activeThreads = at_ptr;
129 }
130 
131 template <class Impl>
132 void
134 {
135  for (ThreadID tid = 0; tid < numThreads; tid++)
136  assert(instList[tid].empty());
137  assert(isEmpty());
138 }
139 
140 template <class Impl>
141 void
143 {
144  resetState();
145 }
146 
147 template <class Impl>
148 void
150 {
151  if (robPolicy != SMTQueuePolicy::Dynamic || numThreads > 1) {
152  auto active_threads = activeThreads->size();
153 
154  std::list<ThreadID>::iterator threads = activeThreads->begin();
155  std::list<ThreadID>::iterator end = activeThreads->end();
156 
157  while (threads != end) {
158  ThreadID tid = *threads++;
159 
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;
165  }
166  }
167  }
168 }
169 
170 template <class Impl>
171 int
173 {
174  if (robPolicy == SMTQueuePolicy::Partitioned) {
175  return numEntries / num_threads;
176  } else {
177  return 0;
178  }
179 }
180 
181 template <class Impl>
182 int
184 {
185  int total = 0;
186 
187  for (ThreadID tid = 0; tid < numThreads; tid++)
188  total += countInsts(tid);
189 
190  return total;
191 }
192 
193 template <class Impl>
194 size_t
196 {
197  return instList[tid].size();
198 }
199 
200 template <class Impl>
201 void
203 {
204  assert(inst);
205 
206  stats.writes++;
207 
208  DPRINTF(ROB, "Adding inst PC %s to the ROB.\n", inst->pcState());
209 
210  assert(numInstsInROB != numEntries);
211 
212  ThreadID tid = inst->threadNumber;
213 
214  instList[tid].push_back(inst);
215 
216  //Set Up head iterator if this is the 1st instruction in the ROB
217  if (numInstsInROB == 0) {
218  head = instList[tid].begin();
219  assert((*head) == inst);
220  }
221 
222  //Must Decrement for iterator to actually be valid since __.end()
223  //actually points to 1 after the last inst
224  tail = instList[tid].end();
225  tail--;
226 
227  inst->setInROB();
228 
229  ++numInstsInROB;
230  ++threadEntries[tid];
231 
232  assert((*tail) == inst);
233 
234  DPRINTF(ROB, "[tid:%i] Now has %d instructions.\n", tid, threadEntries[tid]);
235 }
236 
237 template <class Impl>
238 void
240 {
241  stats.writes++;
242 
243  assert(numInstsInROB > 0);
244 
245  // Get the head ROB instruction by copying it and remove it from the list
246  InstIt head_it = instList[tid].begin();
247 
248  DynInstPtr head_inst = std::move(*head_it);
249  instList[tid].erase(head_it);
250 
251  assert(head_inst->readyToCommit());
252 
253  DPRINTF(ROB, "[tid:%i] Retiring head instruction, "
254  "instruction PC %s, [sn:%llu]\n", tid, head_inst->pcState(),
255  head_inst->seqNum);
256 
257  --numInstsInROB;
258  --threadEntries[tid];
259 
260  head_inst->clearInROB();
261  head_inst->setCommitted();
262 
263  //Update "Global" Head of ROB
264  updateHead();
265 
266  // @todo: A special case is needed if the instruction being
267  // retired is the only instruction in the ROB; otherwise the tail
268  // iterator will become invalidated.
269  cpu->removeFrontInst(head_inst);
270 }
271 
272 template <class Impl>
273 bool
275 {
276  stats.reads++;
277  if (threadEntries[tid] != 0) {
278  return instList[tid].front()->readyToCommit();
279  }
280 
281  return false;
282 }
283 
284 template <class Impl>
285 bool
287 {
288  //@todo: set ActiveThreads through ROB or CPU
289  std::list<ThreadID>::iterator threads = activeThreads->begin();
290  std::list<ThreadID>::iterator end = activeThreads->end();
291 
292  while (threads != end) {
293  ThreadID tid = *threads++;
294 
295  if (isHeadReady(tid)) {
296  return true;
297  }
298  }
299 
300  return false;
301 }
302 
303 template <class Impl>
304 unsigned
306 {
307  return numEntries - numInstsInROB;
308 }
309 
310 template <class Impl>
311 unsigned
313 {
314  return maxEntries[tid] - threadEntries[tid];
315 }
316 
317 template <class Impl>
318 void
320 {
321  stats.writes++;
322  DPRINTF(ROB, "[tid:%i] Squashing instructions until [sn:%llu].\n",
323  tid, squashedSeqNum[tid]);
324 
325  assert(squashIt[tid] != instList[tid].end());
326 
327  if ((*squashIt[tid])->seqNum < squashedSeqNum[tid]) {
328  DPRINTF(ROB, "[tid:%i] Done squashing instructions.\n",
329  tid);
330 
331  squashIt[tid] = instList[tid].end();
332 
333  doneSquashing[tid] = true;
334  return;
335  }
336 
337  bool robTailUpdate = false;
338 
339  unsigned int numInstsToSquash = squashWidth;
340 
341  // If the CPU is exiting, squash all of the instructions
342  // it is told to, even if that exceeds the squashWidth.
343  // Set the number to the number of entries (the max).
344  if (cpu->isThreadExiting(tid))
345  {
346  numInstsToSquash = numEntries;
347  }
348 
349  for (int numSquashed = 0;
350  numSquashed < numInstsToSquash &&
351  squashIt[tid] != instList[tid].end() &&
352  (*squashIt[tid])->seqNum > squashedSeqNum[tid];
353  ++numSquashed)
354  {
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);
359 
360  // Mark the instruction as squashed, and ready to commit so that
361  // it can drain out of the pipeline.
362  (*squashIt[tid])->setSquashed();
363 
364  (*squashIt[tid])->setCanCommit();
365 
366 
367  if (squashIt[tid] == instList[tid].begin()) {
368  DPRINTF(ROB, "Reached head of instruction list while "
369  "squashing.\n");
370 
371  squashIt[tid] = instList[tid].end();
372 
373  doneSquashing[tid] = true;
374 
375  return;
376  }
377 
378  InstIt tail_thread = instList[tid].end();
379  tail_thread--;
380 
381  if ((*squashIt[tid]) == (*tail_thread))
382  robTailUpdate = true;
383 
384  squashIt[tid]--;
385  }
386 
387 
388  // Check if ROB is done squashing.
389  if ((*squashIt[tid])->seqNum <= squashedSeqNum[tid]) {
390  DPRINTF(ROB, "[tid:%i] Done squashing instructions.\n",
391  tid);
392 
393  squashIt[tid] = instList[tid].end();
394 
395  doneSquashing[tid] = true;
396  }
397 
398  if (robTailUpdate) {
399  updateTail();
400  }
401 }
402 
403 
404 template <class Impl>
405 void
407 {
408  InstSeqNum lowest_num = 0;
409  bool first_valid = true;
410 
411  // @todo: set ActiveThreads through ROB or CPU
412  std::list<ThreadID>::iterator threads = activeThreads->begin();
413  std::list<ThreadID>::iterator end = activeThreads->end();
414 
415  while (threads != end) {
416  ThreadID tid = *threads++;
417 
418  if (instList[tid].empty())
419  continue;
420 
421  if (first_valid) {
422  head = instList[tid].begin();
423  lowest_num = (*head)->seqNum;
424  first_valid = false;
425  continue;
426  }
427 
428  InstIt head_thread = instList[tid].begin();
429 
430  DynInstPtr head_inst = (*head_thread);
431 
432  assert(head_inst != 0);
433 
434  if (head_inst->seqNum < lowest_num) {
435  head = head_thread;
436  lowest_num = head_inst->seqNum;
437  }
438  }
439 
440  if (first_valid) {
441  head = instList[0].end();
442  }
443 
444 }
445 
446 template <class Impl>
447 void
449 {
450  tail = instList[0].end();
451  bool first_valid = true;
452 
453  std::list<ThreadID>::iterator threads = activeThreads->begin();
454  std::list<ThreadID>::iterator end = activeThreads->end();
455 
456  while (threads != end) {
457  ThreadID tid = *threads++;
458 
459  if (instList[tid].empty()) {
460  continue;
461  }
462 
463  // If this is the first valid then assign w/out
464  // comparison
465  if (first_valid) {
466  tail = instList[tid].end();
467  tail--;
468  first_valid = false;
469  continue;
470  }
471 
472  // Assign new tail if this thread's tail is younger
473  // than our current "tail high"
474  InstIt tail_thread = instList[tid].end();
475  tail_thread--;
476 
477  if ((*tail_thread)->seqNum > (*tail)->seqNum) {
478  tail = tail_thread;
479  }
480  }
481 }
482 
483 
484 template <class Impl>
485 void
487 {
488  if (isEmpty(tid)) {
489  DPRINTF(ROB, "Does not need to squash due to being empty "
490  "[sn:%llu]\n",
491  squash_num);
492 
493  return;
494  }
495 
496  DPRINTF(ROB, "Starting to squash within the ROB.\n");
497 
498  robStatus[tid] = ROBSquashing;
499 
500  doneSquashing[tid] = false;
501 
502  squashedSeqNum[tid] = squash_num;
503 
504  if (!instList[tid].empty()) {
505  InstIt tail_thread = instList[tid].end();
506  tail_thread--;
507 
508  squashIt[tid] = tail_thread;
509 
510  doSquash(tid);
511  }
512 }
513 
514 template <class Impl>
515 const typename Impl::DynInstPtr&
517 {
518  if (threadEntries[tid] != 0) {
519  InstIt head_thread = instList[tid].begin();
520 
521  assert((*head_thread)->isInROB());
522 
523  return *head_thread;
524  } else {
525  return dummyInst;
526  }
527 }
528 
529 template <class Impl>
530 typename Impl::DynInstPtr
532 {
533  InstIt tail_thread = instList[tid].end();
534  tail_thread--;
535 
536  return *tail_thread;
537 }
538 
539 template <class Impl>
541  : Stats::Group(parent, "rob"),
542  ADD_STAT(reads, UNIT_COUNT, "The number of ROB reads"),
543  ADD_STAT(writes, UNIT_COUNT, "The number of ROB writes")
544 {
545 }
546 
547 template <class Impl>
548 typename Impl::DynInstPtr
550 {
551  for (InstIt it = instList[tid].begin(); it != instList[tid].end(); it++) {
552  if ((*it)->seqNum == squash_inst) {
553  return *it;
554  }
555  }
556  return NULL;
557 }
558 
559 #endif//__CPU_O3_ROB_IMPL_HH__
ROB::resetEntries
void resetEntries()
Re-adjust ROB partitioning.
Definition: rob_impl.hh:149
ROB::O3CPU
Impl::O3CPU O3CPU
Definition: rob.hh:63
ThreadID
int16_t ThreadID
Thread index/ID type.
Definition: types.hh:233
ROB::findInst
DynInstPtr findInst(ThreadID tid, InstSeqNum squash_inst)
Returns a pointer to the instruction with the given sequence if it is in the ROB.
Definition: rob_impl.hh:549
ROB::squash
void squash(InstSeqNum squash_num, ThreadID tid)
Squashes all instructions younger than the given sequence number for the specific thread.
Definition: rob_impl.hh:486
ROB::updateHead
void updateHead()
Updates the head instruction with the new oldest instruction.
Definition: rob_impl.hh:406
ROB::readTailInst
DynInstPtr readTailInst(ThreadID tid)
Returns pointer to the tail instruction within the ROB.
Definition: rob_impl.hh:531
ROB::readHeadInst
const DynInstPtr & readHeadInst(ThreadID tid)
Returns pointer to the head instruction within the ROB.
Definition: rob_impl.hh:516
ROB::name
std::string name() const
Definition: rob_impl.hh:118
ROB::updateTail
void updateTail()
Updates the tail instruction with the new youngest instruction.
Definition: rob_impl.hh:448
ROB::maxEntries
unsigned maxEntries[Impl::MaxThreads]
Max Insts a Thread Can Have in the ROB.
Definition: rob.hh:277
rob.hh
ROB::numThreads
ThreadID numThreads
Number of active threads.
Definition: rob.hh:321
ROB::instList
std::list< DynInstPtr > instList[Impl::MaxThreads]
ROB List of Instructions.
Definition: rob.hh:280
ROB::entryAmount
int entryAmount(ThreadID num_threads)
Number of entries needed For 'num_threads' amount of threads.
Definition: rob_impl.hh:172
DPRINTF
#define DPRINTF(x,...)
Definition: trace.hh:237
ADD_STAT
#define ADD_STAT(n,...)
Convenience macro to add a stat to a statistics group.
Definition: group.hh:71
ROB::takeOverFrom
void takeOverFrom()
Takes over another CPU's thread.
Definition: rob_impl.hh:142
ROB::drainSanityCheck
void drainSanityCheck() const
Perform sanity checks after a drain.
Definition: rob_impl.hh:133
ROB::canCommit
bool canCommit()
Is there any commitable head instruction across all threads ready.
Definition: rob_impl.hh:286
ROB::doSquash
void doSquash(ThreadID tid)
Executes the squash, marking squashed instructions.
Definition: rob_impl.hh:319
ROB::ROBStats::ROBStats
ROBStats(Stats::Group *parent)
Definition: rob_impl.hh:540
InstSeqNum
uint64_t InstSeqNum
Definition: inst_seq.hh:37
UNIT_COUNT
#define UNIT_COUNT
Definition: units.hh:49
ROB::isHeadReady
bool isHeadReady(ThreadID tid)
Is the oldest instruction across all threads ready.
Definition: rob_impl.hh:274
ROB::resetState
void resetState()
Reset the ROB state.
Definition: rob_impl.hh:100
ROB::DynInstPtr
Impl::DynInstPtr DynInstPtr
Definition: rob.hh:64
ROB::InstIt
std::list< DynInstPtr >::iterator InstIt
Definition: rob.hh:67
ROB::retireHead
void retireHead(ThreadID tid)
Retires the head instruction, removing it from the ROB.
Definition: rob_impl.hh:239
Stats::Group
Statistics container.
Definition: group.hh:87
ROB::numEntries
unsigned numEntries
Number of instructions in the ROB.
Definition: rob.hh:271
logging.hh
ROB::countInsts
int countInsts()
This is more of a debugging function than anything.
Definition: rob_impl.hh:183
Stats
Definition: statistics.cc:53
std::list< ThreadID >
Stats::total
const FlagsType total
Print the total.
Definition: info.hh:50
ROB::ROB
ROB(O3CPU *_cpu, const DerivO3CPUParams &params)
ROB constructor.
Definition: rob_impl.hh:53
ROB::numFreeEntries
unsigned numFreeEntries()
Returns the number of total free entries in the ROB.
Definition: rob_impl.hh:305
ROB
ROB class.
Definition: rob.hh:59
ROB::robPolicy
SMTQueuePolicy robPolicy
ROB resource sharing policy for SMT mode.
Definition: rob.hh:81
ROB::insertInst
void insertInst(const DynInstPtr &inst)
Function to insert an instruction into the ROB.
Definition: rob_impl.hh:202
ROB::setActiveThreads
void setActiveThreads(std::list< ThreadID > *at_ptr)
Sets pointer to the list of active threads.
Definition: rob_impl.hh:125

Generated on Tue Jun 22 2021 15:28:26 for gem5 by doxygen 1.8.17