gem5  v21.2.1.1
rename.cc
Go to the documentation of this file.
1 /*
2  * Copyright (c) 2010-2012, 2014-2019 ARM Limited
3  * Copyright (c) 2013 Advanced Micro Devices, Inc.
4  * All rights reserved.
5  *
6  * The license below extends only to copyright in the software and shall
7  * not be construed as granting a license to any other intellectual
8  * property including but not limited to intellectual property relating
9  * to a hardware implementation of the functionality of the software
10  * licensed hereunder. You may use the software subject to the license
11  * terms below provided that you ensure that this notice is replicated
12  * unmodified and in its entirety in all distributions of the software,
13  * modified or unmodified, in source code or in binary form.
14  *
15  * Copyright (c) 2004-2006 The Regents of The University of Michigan
16  * All rights reserved.
17  *
18  * Redistribution and use in source and binary forms, with or without
19  * modification, are permitted provided that the following conditions are
20  * met: redistributions of source code must retain the above copyright
21  * notice, this list of conditions and the following disclaimer;
22  * redistributions in binary form must reproduce the above copyright
23  * notice, this list of conditions and the following disclaimer in the
24  * documentation and/or other materials provided with the distribution;
25  * neither the name of the copyright holders nor the names of its
26  * contributors may be used to endorse or promote products derived from
27  * this software without specific prior written permission.
28  *
29  * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
30  * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
31  * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
32  * A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
33  * OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
34  * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
35  * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
36  * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
37  * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
38  * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
39  * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
40  */
41 
42 #include "cpu/o3/rename.hh"
43 
44 #include <list>
45 
46 #include "cpu/o3/cpu.hh"
47 #include "cpu/o3/dyn_inst.hh"
48 #include "cpu/o3/limits.hh"
49 #include "cpu/reg_class.hh"
50 #include "debug/Activity.hh"
51 #include "debug/O3PipeView.hh"
52 #include "debug/Rename.hh"
53 #include "params/O3CPU.hh"
54 
55 namespace gem5
56 {
57 
58 namespace o3
59 {
60 
61 Rename::Rename(CPU *_cpu, const O3CPUParams &params)
62  : cpu(_cpu),
63  iewToRenameDelay(params.iewToRenameDelay),
64  decodeToRenameDelay(params.decodeToRenameDelay),
65  commitToRenameDelay(params.commitToRenameDelay),
66  renameWidth(params.renameWidth),
67  numThreads(params.numThreads),
68  stats(_cpu)
69 {
70  if (renameWidth > MaxWidth)
71  fatal("renameWidth (%d) is larger than compiled limit (%d),\n"
72  "\tincrease MaxWidth in src/cpu/o3/limits.hh\n",
73  renameWidth, static_cast<int>(MaxWidth));
74 
75  // @todo: Make into a parameter.
76  skidBufferMax = (decodeToRenameDelay + 1) * params.decodeWidth;
77  for (uint32_t tid = 0; tid < MaxThreads; tid++) {
78  renameStatus[tid] = Idle;
79  renameMap[tid] = nullptr;
80  instsInProgress[tid] = 0;
81  loadsInProgress[tid] = 0;
82  storesInProgress[tid] = 0;
83  freeEntries[tid] = {0, 0, 0, 0};
84  emptyROB[tid] = true;
85  stalls[tid] = {false, false};
86  serializeInst[tid] = nullptr;
87  serializeOnNextInst[tid] = false;
88  }
89 }
90 
91 std::string
92 Rename::name() const
93 {
94  return cpu->name() + ".rename";
95 }
96 
98  : statistics::Group(parent, "rename"),
99  ADD_STAT(squashCycles, statistics::units::Cycle::get(),
100  "Number of cycles rename is squashing"),
101  ADD_STAT(idleCycles, statistics::units::Cycle::get(),
102  "Number of cycles rename is idle"),
103  ADD_STAT(blockCycles, statistics::units::Cycle::get(),
104  "Number of cycles rename is blocking"),
105  ADD_STAT(serializeStallCycles, statistics::units::Cycle::get(),
106  "count of cycles rename stalled for serializing inst"),
107  ADD_STAT(runCycles, statistics::units::Cycle::get(),
108  "Number of cycles rename is running"),
109  ADD_STAT(unblockCycles, statistics::units::Cycle::get(),
110  "Number of cycles rename is unblocking"),
111  ADD_STAT(renamedInsts, statistics::units::Count::get(),
112  "Number of instructions processed by rename"),
113  ADD_STAT(squashedInsts, statistics::units::Count::get(),
114  "Number of squashed instructions processed by rename"),
115  ADD_STAT(ROBFullEvents, statistics::units::Count::get(),
116  "Number of times rename has blocked due to ROB full"),
117  ADD_STAT(IQFullEvents, statistics::units::Count::get(),
118  "Number of times rename has blocked due to IQ full"),
119  ADD_STAT(LQFullEvents, statistics::units::Count::get(),
120  "Number of times rename has blocked due to LQ full" ),
121  ADD_STAT(SQFullEvents, statistics::units::Count::get(),
122  "Number of times rename has blocked due to SQ full"),
123  ADD_STAT(fullRegistersEvents, statistics::units::Count::get(),
124  "Number of times there has been no free registers"),
125  ADD_STAT(renamedOperands, statistics::units::Count::get(),
126  "Number of destination operands rename has renamed"),
127  ADD_STAT(lookups, statistics::units::Count::get(),
128  "Number of register rename lookups that rename has made"),
129  ADD_STAT(intLookups, statistics::units::Count::get(),
130  "Number of integer rename lookups"),
131  ADD_STAT(fpLookups, statistics::units::Count::get(),
132  "Number of floating rename lookups"),
133  ADD_STAT(vecLookups, statistics::units::Count::get(),
134  "Number of vector rename lookups"),
135  ADD_STAT(vecPredLookups, statistics::units::Count::get(),
136  "Number of vector predicate rename lookups"),
137  ADD_STAT(committedMaps, statistics::units::Count::get(),
138  "Number of HB maps that are committed"),
139  ADD_STAT(undoneMaps, statistics::units::Count::get(),
140  "Number of HB maps that are undone due to squashing"),
141  ADD_STAT(serializing, statistics::units::Count::get(),
142  "count of serializing insts renamed"),
143  ADD_STAT(tempSerializing, statistics::units::Count::get(),
144  "count of temporary serializing insts renamed"),
145  ADD_STAT(skidInsts, statistics::units::Count::get(),
146  "count of insts added to the skid buffer")
147 {
154 
157 
163 
170 
176 }
177 
178 void
180 {
182  cpu->getProbeManager(), "Rename");
183  ppSquashInRename = new ProbePointArg<SeqNumRegPair>(cpu->getProbeManager(),
184  "SquashInRename");
185 }
186 
187 void
189 {
190  timeBuffer = tb_ptr;
191 
192  // Setup wire to read information from time buffer, from IEW stage.
193  fromIEW = timeBuffer->getWire(-iewToRenameDelay);
194 
195  // Setup wire to read infromation from time buffer, from commit stage.
197 
198  // Setup wire to write information to previous stages.
199  toDecode = timeBuffer->getWire(0);
200 }
201 
202 void
204 {
205  renameQueue = rq_ptr;
206 
207  // Setup wire to write information to future stages.
208  toIEW = renameQueue->getWire(0);
209 }
210 
211 void
213 {
214  decodeQueue = dq_ptr;
215 
216  // Setup wire to get information from decode.
218 }
219 
220 void
222 {
223  resetStage();
224 }
225 
226 void
228 {
229  renameStatus[tid] = Idle;
230 
235  emptyROB[tid] = true;
236 
237  stalls[tid].iew = false;
238  serializeInst[tid] = NULL;
239 
240  instsInProgress[tid] = 0;
241  loadsInProgress[tid] = 0;
242  storesInProgress[tid] = 0;
243 
244  serializeOnNextInst[tid] = false;
245 }
246 
247 void
249 {
250  _status = Inactive;
251 
252  resumeSerialize = false;
253  resumeUnblocking = false;
254 
255  // Grab the number of free entries directly from the stages.
256  for (ThreadID tid = 0; tid < numThreads; tid++) {
257  renameStatus[tid] = Idle;
258 
260  freeEntries[tid].lqEntries =
262  freeEntries[tid].sqEntries =
265  emptyROB[tid] = true;
266 
267  stalls[tid].iew = false;
268  serializeInst[tid] = NULL;
269 
270  instsInProgress[tid] = 0;
271  loadsInProgress[tid] = 0;
272  storesInProgress[tid] = 0;
273 
274  serializeOnNextInst[tid] = false;
275  }
276 }
277 
278 void
280 {
281  activeThreads = at_ptr;
282 }
283 
284 
285 void
287 {
288  for (ThreadID tid = 0; tid < numThreads; tid++)
289  renameMap[tid] = &rm_ptr[tid];
290 }
291 
292 void
294 {
295  freeList = fl_ptr;
296 }
297 
298 void
300 {
301  scoreboard = _scoreboard;
302 }
303 
304 bool
306 {
307  for (ThreadID tid = 0; tid < numThreads; tid++) {
308  if (instsInProgress[tid] != 0 ||
309  !historyBuffer[tid].empty() ||
310  !skidBuffer[tid].empty() ||
311  !insts[tid].empty() ||
312  (renameStatus[tid] != Idle && renameStatus[tid] != Running))
313  return false;
314  }
315  return true;
316 }
317 
318 void
320 {
321  resetStage();
322 }
323 
324 void
326 {
327  for (ThreadID tid = 0; tid < numThreads; tid++) {
328  assert(historyBuffer[tid].empty());
329  assert(insts[tid].empty());
330  assert(skidBuffer[tid].empty());
331  assert(instsInProgress[tid] == 0);
332  }
333 }
334 
335 void
336 Rename::squash(const InstSeqNum &squash_seq_num, ThreadID tid)
337 {
338  DPRINTF(Rename, "[tid:%i] [squash sn:%llu] Squashing instructions.\n",
339  tid,squash_seq_num);
340 
341  // Clear the stall signal if rename was blocked or unblocking before.
342  // If it still needs to block, the blocking should happen the next
343  // cycle and there should be space to hold everything due to the squash.
344  if (renameStatus[tid] == Blocked ||
345  renameStatus[tid] == Unblocking) {
346  toDecode->renameUnblock[tid] = 1;
347 
348  resumeSerialize = false;
349  serializeInst[tid] = NULL;
350  } else if (renameStatus[tid] == SerializeStall) {
351  if (serializeInst[tid]->seqNum <= squash_seq_num) {
352  DPRINTF(Rename, "[tid:%i] [squash sn:%llu] "
353  "Rename will resume serializing after squash\n",
354  tid,squash_seq_num);
355  resumeSerialize = true;
356  assert(serializeInst[tid]);
357  } else {
358  resumeSerialize = false;
359  toDecode->renameUnblock[tid] = 1;
360 
361  serializeInst[tid] = NULL;
362  }
363  }
364 
365  // Set the status to Squashing.
366  renameStatus[tid] = Squashing;
367 
368  // Squash any instructions from decode.
369  for (int i=0; i<fromDecode->size; i++) {
370  if (fromDecode->insts[i]->threadNumber == tid &&
371  fromDecode->insts[i]->seqNum > squash_seq_num) {
372  fromDecode->insts[i]->setSquashed();
373  wroteToTimeBuffer = true;
374  }
375 
376  }
377 
378  // Clear the instruction list and skid buffer in case they have any
379  // insts in them.
380  insts[tid].clear();
381 
382  // Clear the skid buffer in case it has any data in it.
383  skidBuffer[tid].clear();
384 
385  doSquash(squash_seq_num, tid);
386 }
387 
388 void
390 {
391  wroteToTimeBuffer = false;
392 
393  blockThisCycle = false;
394 
395  bool status_change = false;
396 
397  toIEWIndex = 0;
398 
399  sortInsts();
400 
401  std::list<ThreadID>::iterator threads = activeThreads->begin();
403 
404  // Check stall and squash signals.
405  while (threads != end) {
406  ThreadID tid = *threads++;
407 
408  DPRINTF(Rename, "Processing [tid:%i]\n", tid);
409 
410  status_change = checkSignalsAndUpdate(tid) || status_change;
411 
412  rename(status_change, tid);
413  }
414 
415  if (status_change) {
416  updateStatus();
417  }
418 
419  if (wroteToTimeBuffer) {
420  DPRINTF(Activity, "Activity this cycle.\n");
422  }
423 
424  threads = activeThreads->begin();
425 
426  while (threads != end) {
427  ThreadID tid = *threads++;
428 
429  // If we committed this cycle then doneSeqNum will be > 0
430  if (fromCommit->commitInfo[tid].doneSeqNum != 0 &&
431  !fromCommit->commitInfo[tid].squash &&
432  renameStatus[tid] != Squashing) {
433 
434  removeFromHistory(fromCommit->commitInfo[tid].doneSeqNum,
435  tid);
436  }
437  }
438 
439  // @todo: make into updateProgress function
440  for (ThreadID tid = 0; tid < numThreads; tid++) {
441  instsInProgress[tid] -= fromIEW->iewInfo[tid].dispatched;
442  loadsInProgress[tid] -= fromIEW->iewInfo[tid].dispatchedToLQ;
443  storesInProgress[tid] -= fromIEW->iewInfo[tid].dispatchedToSQ;
444  assert(loadsInProgress[tid] >= 0);
445  assert(storesInProgress[tid] >= 0);
446  assert(instsInProgress[tid] >=0);
447  }
448 
449 }
450 
451 void
452 Rename::rename(bool &status_change, ThreadID tid)
453 {
454  // If status is Running or idle,
455  // call renameInsts()
456  // If status is Unblocking,
457  // buffer any instructions coming from decode
458  // continue trying to empty skid buffer
459  // check if stall conditions have passed
460 
461  if (renameStatus[tid] == Blocked) {
462  ++stats.blockCycles;
463  } else if (renameStatus[tid] == Squashing) {
465  } else if (renameStatus[tid] == SerializeStall) {
467  // If we are currently in SerializeStall and resumeSerialize
468  // was set, then that means that we are resuming serializing
469  // this cycle. Tell the previous stages to block.
470  if (resumeSerialize) {
471  resumeSerialize = false;
472  block(tid);
473  toDecode->renameUnblock[tid] = false;
474  }
475  } else if (renameStatus[tid] == Unblocking) {
476  if (resumeUnblocking) {
477  block(tid);
478  resumeUnblocking = false;
479  toDecode->renameUnblock[tid] = false;
480  }
481  }
482 
483  if (renameStatus[tid] == Running ||
484  renameStatus[tid] == Idle) {
485  DPRINTF(Rename,
486  "[tid:%i] "
487  "Not blocked, so attempting to run stage.\n",
488  tid);
489 
490  renameInsts(tid);
491  } else if (renameStatus[tid] == Unblocking) {
492  renameInsts(tid);
493 
494  if (validInsts()) {
495  // Add the current inputs to the skid buffer so they can be
496  // reprocessed when this stage unblocks.
497  skidInsert(tid);
498  }
499 
500  // If we switched over to blocking, then there's a potential for
501  // an overall status change.
502  status_change = unblock(tid) || status_change || blockThisCycle;
503  }
504 }
505 
506 void
508 {
509  // Instructions can be either in the skid buffer or the queue of
510  // instructions coming from decode, depending on the status.
511  int insts_available = renameStatus[tid] == Unblocking ?
512  skidBuffer[tid].size() : insts[tid].size();
513 
514  // Check the decode queue to see if instructions are available.
515  // If there are no available instructions to rename, then do nothing.
516  if (insts_available == 0) {
517  DPRINTF(Rename, "[tid:%i] Nothing to do, breaking out early.\n",
518  tid);
519  // Should I change status to idle?
520  ++stats.idleCycles;
521  return;
522  } else if (renameStatus[tid] == Unblocking) {
524  } else if (renameStatus[tid] == Running) {
525  ++stats.runCycles;
526  }
527 
528  // Will have to do a different calculation for the number of free
529  // entries.
530  int free_rob_entries = calcFreeROBEntries(tid);
531  int free_iq_entries = calcFreeIQEntries(tid);
532  int min_free_entries = free_rob_entries;
533 
534  FullSource source = ROB;
535 
536  if (free_iq_entries < min_free_entries) {
537  min_free_entries = free_iq_entries;
538  source = IQ;
539  }
540 
541  // Check if there's any space left.
542  if (min_free_entries <= 0) {
543  DPRINTF(Rename,
544  "[tid:%i] Blocking due to no free ROB/IQ/ entries.\n"
545  "ROB has %i free entries.\n"
546  "IQ has %i free entries.\n",
547  tid, free_rob_entries, free_iq_entries);
548 
549  blockThisCycle = true;
550 
551  block(tid);
552 
553  incrFullStat(source);
554 
555  return;
556  } else if (min_free_entries < insts_available) {
557  DPRINTF(Rename,
558  "[tid:%i] "
559  "Will have to block this cycle. "
560  "%i insts available, "
561  "but only %i insts can be renamed due to ROB/IQ/LSQ limits.\n",
562  tid, insts_available, min_free_entries);
563 
564  insts_available = min_free_entries;
565 
566  blockThisCycle = true;
567 
568  incrFullStat(source);
569  }
570 
571  InstQueue &insts_to_rename = renameStatus[tid] == Unblocking ?
572  skidBuffer[tid] : insts[tid];
573 
574  DPRINTF(Rename,
575  "[tid:%i] "
576  "%i available instructions to send iew.\n",
577  tid, insts_available);
578 
579  DPRINTF(Rename,
580  "[tid:%i] "
581  "%i insts pipelining from Rename | "
582  "%i insts dispatched to IQ last cycle.\n",
583  tid, instsInProgress[tid], fromIEW->iewInfo[tid].dispatched);
584 
585  // Handle serializing the next instruction if necessary.
586  if (serializeOnNextInst[tid]) {
587  if (emptyROB[tid] && instsInProgress[tid] == 0) {
588  // ROB already empty; no need to serialize.
589  serializeOnNextInst[tid] = false;
590  } else if (!insts_to_rename.empty()) {
591  insts_to_rename.front()->setSerializeBefore();
592  }
593  }
594 
595  int renamed_insts = 0;
596 
597  while (insts_available > 0 && toIEWIndex < renameWidth) {
598  DPRINTF(Rename, "[tid:%i] Sending instructions to IEW.\n", tid);
599 
600  assert(!insts_to_rename.empty());
601 
602  DynInstPtr inst = insts_to_rename.front();
603 
604  //For all kind of instructions, check ROB and IQ first For load
605  //instruction, check LQ size and take into account the inflight loads
606  //For store instruction, check SQ size and take into account the
607  //inflight stores
608 
609  if (inst->isLoad()) {
610  if (calcFreeLQEntries(tid) <= 0) {
611  DPRINTF(Rename, "[tid:%i] Cannot rename due to no free LQ\n",
612  tid);
613  source = LQ;
614  incrFullStat(source);
615  break;
616  }
617  }
618 
619  if (inst->isStore() || inst->isAtomic()) {
620  if (calcFreeSQEntries(tid) <= 0) {
621  DPRINTF(Rename, "[tid:%i] Cannot rename due to no free SQ\n",
622  tid);
623  source = SQ;
624  incrFullStat(source);
625  break;
626  }
627  }
628 
629  insts_to_rename.pop_front();
630 
631  if (renameStatus[tid] == Unblocking) {
632  DPRINTF(Rename,
633  "[tid:%i] "
634  "Removing [sn:%llu] PC:%s from rename skidBuffer\n",
635  tid, inst->seqNum, inst->pcState());
636  }
637 
638  if (inst->isSquashed()) {
639  DPRINTF(Rename,
640  "[tid:%i] "
641  "instruction %i with PC %s is squashed, skipping.\n",
642  tid, inst->seqNum, inst->pcState());
643 
645 
646  // Decrement how many instructions are available.
647  --insts_available;
648 
649  continue;
650  }
651 
652  DPRINTF(Rename,
653  "[tid:%i] "
654  "Processing instruction [sn:%llu] with PC %s.\n",
655  tid, inst->seqNum, inst->pcState());
656 
657  // Check here to make sure there are enough destination registers
658  // to rename to. Otherwise block.
659  if (!renameMap[tid]->canRename(inst->numIntDestRegs(),
660  inst->numFPDestRegs(),
661  inst->numVecDestRegs(),
662  inst->numVecElemDestRegs(),
663  inst->numVecPredDestRegs(),
664  inst->numCCDestRegs())) {
665  DPRINTF(Rename,
666  "Blocking due to "
667  " lack of free physical registers to rename to.\n");
668  blockThisCycle = true;
669  insts_to_rename.push_front(inst);
671 
672  break;
673  }
674 
675  // Handle serializeAfter/serializeBefore instructions.
676  // serializeAfter marks the next instruction as serializeBefore.
677  // serializeBefore makes the instruction wait in rename until the ROB
678  // is empty.
679 
680  // In this model, IPR accesses are serialize before
681  // instructions, and store conditionals are serialize after
682  // instructions. This is mainly due to lack of support for
683  // out-of-order operations of either of those classes of
684  // instructions.
685  if (inst->isSerializeBefore() && !inst->isSerializeHandled()) {
686  DPRINTF(Rename, "Serialize before instruction encountered.\n");
687 
688  if (!inst->isTempSerializeBefore()) {
689  stats.serializing++;
690  inst->setSerializeHandled();
691  } else {
693  }
694 
695  // Change status over to SerializeStall so that other stages know
696  // what this is blocked on.
698 
699  serializeInst[tid] = inst;
700 
701  blockThisCycle = true;
702 
703  break;
704  } else if ((inst->isStoreConditional() || inst->isSerializeAfter()) &&
705  !inst->isSerializeHandled()) {
706  DPRINTF(Rename, "Serialize after instruction encountered.\n");
707 
708  stats.serializing++;
709 
710  inst->setSerializeHandled();
711 
712  serializeAfter(insts_to_rename, tid);
713  }
714 
715  renameSrcRegs(inst, inst->threadNumber);
716 
717  renameDestRegs(inst, inst->threadNumber);
718 
719  if (inst->isAtomic() || inst->isStore()) {
720  storesInProgress[tid]++;
721  } else if (inst->isLoad()) {
722  loadsInProgress[tid]++;
723  }
724 
725  ++renamed_insts;
726  // Notify potential listeners that source and destination registers for
727  // this instruction have been renamed.
728  ppRename->notify(inst);
729 
730  // Put instruction in rename queue.
731  toIEW->insts[toIEWIndex] = inst;
732  ++(toIEW->size);
733 
734  // Increment which instruction we're on.
735  ++toIEWIndex;
736 
737  // Decrement how many instructions are available.
738  --insts_available;
739  }
740 
741  instsInProgress[tid] += renamed_insts;
742  stats.renamedInsts += renamed_insts;
743 
744  // If we wrote to the time buffer, record this.
745  if (toIEWIndex) {
746  wroteToTimeBuffer = true;
747  }
748 
749  // Check if there's any instructions left that haven't yet been renamed.
750  // If so then block.
751  if (insts_available) {
752  blockThisCycle = true;
753  }
754 
755  if (blockThisCycle) {
756  block(tid);
757  toDecode->renameUnblock[tid] = false;
758  }
759 }
760 
761 void
763 {
764  DynInstPtr inst = NULL;
765 
766  while (!insts[tid].empty()) {
767  inst = insts[tid].front();
768 
769  insts[tid].pop_front();
770 
771  assert(tid == inst->threadNumber);
772 
773  DPRINTF(Rename, "[tid:%i] Inserting [sn:%llu] PC: %s into Rename "
774  "skidBuffer\n", tid, inst->seqNum, inst->pcState());
775 
776  ++stats.skidInsts;
777 
778  skidBuffer[tid].push_back(inst);
779  }
780 
781  if (skidBuffer[tid].size() > skidBufferMax) {
782  InstQueue::iterator it;
783  warn("Skidbuffer contents:\n");
784  for (it = skidBuffer[tid].begin(); it != skidBuffer[tid].end(); it++) {
785  warn("[tid:%i] %s [sn:%llu].\n", tid,
786  (*it)->staticInst->disassemble(
787  inst->pcState().instAddr()),
788  (*it)->seqNum);
789  }
790  panic("Skidbuffer Exceeded Max Size");
791  }
792 }
793 
794 void
796 {
797  int insts_from_decode = fromDecode->size;
798  for (int i = 0; i < insts_from_decode; ++i) {
799  const DynInstPtr &inst = fromDecode->insts[i];
800  insts[inst->threadNumber].push_back(inst);
801 #if TRACING_ON
802  if (debug::O3PipeView) {
803  inst->renameTick = curTick() - inst->fetchTick;
804  }
805 #endif
806  }
807 }
808 
809 bool
811 {
812  std::list<ThreadID>::iterator threads = activeThreads->begin();
814 
815  while (threads != end) {
816  ThreadID tid = *threads++;
817 
818  if (!skidBuffer[tid].empty())
819  return false;
820  }
821 
822  return true;
823 }
824 
825 void
827 {
828  bool any_unblocking = false;
829 
830  std::list<ThreadID>::iterator threads = activeThreads->begin();
832 
833  while (threads != end) {
834  ThreadID tid = *threads++;
835 
836  if (renameStatus[tid] == Unblocking) {
837  any_unblocking = true;
838  break;
839  }
840  }
841 
842  // Rename will have activity if it's unblocking.
843  if (any_unblocking) {
844  if (_status == Inactive) {
845  _status = Active;
846 
847  DPRINTF(Activity, "Activating stage.\n");
848 
850  }
851  } else {
852  // If it's not unblocking, then rename will not have any internal
853  // activity. Switch it to inactive.
854  if (_status == Active) {
855  _status = Inactive;
856  DPRINTF(Activity, "Deactivating stage.\n");
857 
859  }
860  }
861 }
862 
863 bool
865 {
866  DPRINTF(Rename, "[tid:%i] Blocking.\n", tid);
867 
868  // Add the current inputs onto the skid buffer, so they can be
869  // reprocessed when this stage unblocks.
870  skidInsert(tid);
871 
872  // Only signal backwards to block if the previous stages do not think
873  // rename is already blocked.
874  if (renameStatus[tid] != Blocked) {
875  // If resumeUnblocking is set, we unblocked during the squash,
876  // but now we're have unblocking status. We need to tell earlier
877  // stages to block.
878  if (resumeUnblocking || renameStatus[tid] != Unblocking) {
879  toDecode->renameBlock[tid] = true;
880  toDecode->renameUnblock[tid] = false;
881  wroteToTimeBuffer = true;
882  }
883 
884  // Rename can not go from SerializeStall to Blocked, otherwise
885  // it would not know to complete the serialize stall.
886  if (renameStatus[tid] != SerializeStall) {
887  // Set status to Blocked.
888  renameStatus[tid] = Blocked;
889  return true;
890  }
891  }
892 
893  return false;
894 }
895 
896 bool
898 {
899  DPRINTF(Rename, "[tid:%i] Trying to unblock.\n", tid);
900 
901  // Rename is done unblocking if the skid buffer is empty.
902  if (skidBuffer[tid].empty() && renameStatus[tid] != SerializeStall) {
903 
904  DPRINTF(Rename, "[tid:%i] Done unblocking.\n", tid);
905 
906  toDecode->renameUnblock[tid] = true;
907  wroteToTimeBuffer = true;
908 
909  renameStatus[tid] = Running;
910  return true;
911  }
912 
913  return false;
914 }
915 
916 void
917 Rename::doSquash(const InstSeqNum &squashed_seq_num, ThreadID tid)
918 {
919  auto hb_it = historyBuffer[tid].begin();
920 
921  // After a syscall squashes everything, the history buffer may be empty
922  // but the ROB may still be squashing instructions.
923  // Go through the most recent instructions, undoing the mappings
924  // they did and freeing up the registers.
925  while (!historyBuffer[tid].empty() &&
926  hb_it->instSeqNum > squashed_seq_num) {
927  assert(hb_it != historyBuffer[tid].end());
928 
929  DPRINTF(Rename, "[tid:%i] Removing history entry with sequence "
930  "number %i (archReg: %d, newPhysReg: %d, prevPhysReg: %d).\n",
931  tid, hb_it->instSeqNum, hb_it->archReg.index(),
932  hb_it->newPhysReg->index(), hb_it->prevPhysReg->index());
933 
934  // Undo the rename mapping only if it was really a change.
935  // Special regs that are not really renamed (like misc regs
936  // and the zero reg) can be recognized because the new mapping
937  // is the same as the old one. While it would be merely a
938  // waste of time to update the rename table, we definitely
939  // don't want to put these on the free list.
940  if (hb_it->newPhysReg != hb_it->prevPhysReg) {
941  // Tell the rename map to set the architected register to the
942  // previous physical register that it was renamed to.
943  renameMap[tid]->setEntry(hb_it->archReg, hb_it->prevPhysReg);
944 
945  // Put the renamed physical register back on the free list.
946  freeList->addReg(hb_it->newPhysReg);
947  }
948 
949  // Notify potential listeners that the register mapping needs to be
950  // removed because the instruction it was mapped to got squashed. Note
951  // that this is done before hb_it is incremented.
952  ppSquashInRename->notify(std::make_pair(hb_it->instSeqNum,
953  hb_it->newPhysReg));
954 
955  historyBuffer[tid].erase(hb_it++);
956 
957  ++stats.undoneMaps;
958  }
959 }
960 
961 void
963 {
964  DPRINTF(Rename, "[tid:%i] Removing a committed instruction from the "
965  "history buffer %u (size=%i), until [sn:%llu].\n",
966  tid, tid, historyBuffer[tid].size(), inst_seq_num);
967 
968  auto hb_it = historyBuffer[tid].end();
969 
970  --hb_it;
971 
972  if (historyBuffer[tid].empty()) {
973  DPRINTF(Rename, "[tid:%i] History buffer is empty.\n", tid);
974  return;
975  } else if (hb_it->instSeqNum > inst_seq_num) {
976  DPRINTF(Rename, "[tid:%i] [sn:%llu] "
977  "Old sequence number encountered. "
978  "Ensure that a syscall happened recently.\n",
979  tid,inst_seq_num);
980  return;
981  }
982 
983  // Commit all the renames up until (and including) the committed sequence
984  // number. Some or even all of the committed instructions may not have
985  // rename histories if they did not have destination registers that were
986  // renamed.
987  while (!historyBuffer[tid].empty() &&
988  hb_it != historyBuffer[tid].end() &&
989  hb_it->instSeqNum <= inst_seq_num) {
990 
991  DPRINTF(Rename, "[tid:%i] Freeing up older rename of reg %i (%s), "
992  "[sn:%llu].\n",
993  tid, hb_it->prevPhysReg->index(),
994  hb_it->prevPhysReg->className(),
995  hb_it->instSeqNum);
996 
997  // Don't free special phys regs like misc and zero regs, which
998  // can be recognized because the new mapping is the same as
999  // the old one.
1000  if (hb_it->newPhysReg != hb_it->prevPhysReg) {
1001  freeList->addReg(hb_it->prevPhysReg);
1002  }
1003 
1004  ++stats.committedMaps;
1005 
1006  historyBuffer[tid].erase(hb_it--);
1007  }
1008 }
1009 
1010 void
1012 {
1013  gem5::ThreadContext *tc = inst->tcBase();
1014  UnifiedRenameMap *map = renameMap[tid];
1015  unsigned num_src_regs = inst->numSrcRegs();
1016 
1017  // Get the architectual register numbers from the source and
1018  // operands, and redirect them to the right physical register.
1019  for (int src_idx = 0; src_idx < num_src_regs; src_idx++) {
1020  const RegId& src_reg = inst->srcRegIdx(src_idx);
1021  PhysRegIdPtr renamed_reg;
1022 
1023  renamed_reg = map->lookup(tc->flattenRegId(src_reg));
1024  switch (src_reg.classValue()) {
1025  case IntRegClass:
1026  stats.intLookups++;
1027  break;
1028  case FloatRegClass:
1029  stats.fpLookups++;
1030  break;
1031  case VecRegClass:
1032  case VecElemClass:
1033  stats.vecLookups++;
1034  break;
1035  case VecPredRegClass:
1037  break;
1038  case CCRegClass:
1039  case MiscRegClass:
1040  break;
1041 
1042  default:
1043  panic("Invalid register class: %d.", src_reg.classValue());
1044  }
1045 
1046  DPRINTF(Rename,
1047  "[tid:%i] "
1048  "Looking up %s arch reg %i, got phys reg %i (%s)\n",
1049  tid, src_reg.className(),
1050  src_reg.index(), renamed_reg->index(),
1051  renamed_reg->className());
1052 
1053  inst->renameSrcReg(src_idx, renamed_reg);
1054 
1055  // See if the register is ready or not.
1056  if (scoreboard->getReg(renamed_reg)) {
1057  DPRINTF(Rename,
1058  "[tid:%i] "
1059  "Register %d (flat: %d) (%s) is ready.\n",
1060  tid, renamed_reg->index(), renamed_reg->flatIndex(),
1061  renamed_reg->className());
1062 
1063  inst->markSrcRegReady(src_idx);
1064  } else {
1065  DPRINTF(Rename,
1066  "[tid:%i] "
1067  "Register %d (flat: %d) (%s) is not ready.\n",
1068  tid, renamed_reg->index(), renamed_reg->flatIndex(),
1069  renamed_reg->className());
1070  }
1071 
1072  ++stats.lookups;
1073  }
1074 }
1075 
1076 void
1078 {
1079  gem5::ThreadContext *tc = inst->tcBase();
1080  UnifiedRenameMap *map = renameMap[tid];
1081  unsigned num_dest_regs = inst->numDestRegs();
1082 
1083  // Rename the destination registers.
1084  for (int dest_idx = 0; dest_idx < num_dest_regs; dest_idx++) {
1085  const RegId& dest_reg = inst->destRegIdx(dest_idx);
1086  UnifiedRenameMap::RenameInfo rename_result;
1087 
1088  RegId flat_dest_regid = tc->flattenRegId(dest_reg);
1089  flat_dest_regid.setNumPinnedWrites(dest_reg.getNumPinnedWrites());
1090 
1091  rename_result = map->rename(flat_dest_regid);
1092 
1093  inst->flattenedDestIdx(dest_idx, flat_dest_regid);
1094 
1095  scoreboard->unsetReg(rename_result.first);
1096 
1097  DPRINTF(Rename,
1098  "[tid:%i] "
1099  "Renaming arch reg %i (%s) to physical reg %i (%i).\n",
1100  tid, dest_reg.index(), dest_reg.className(),
1101  rename_result.first->index(),
1102  rename_result.first->flatIndex());
1103 
1104  // Record the rename information so that a history can be kept.
1105  RenameHistory hb_entry(inst->seqNum, flat_dest_regid,
1106  rename_result.first,
1107  rename_result.second);
1108 
1109  historyBuffer[tid].push_front(hb_entry);
1110 
1111  DPRINTF(Rename, "[tid:%i] [sn:%llu] "
1112  "Adding instruction to history buffer (size=%i).\n",
1113  tid,(*historyBuffer[tid].begin()).instSeqNum,
1114  historyBuffer[tid].size());
1115 
1116  // Tell the instruction to rename the appropriate destination
1117  // register (dest_idx) to the new physical register
1118  // (rename_result.first), and record the previous physical
1119  // register that the same logical register was renamed to
1120  // (rename_result.second).
1121  inst->renameDestReg(dest_idx,
1122  rename_result.first,
1123  rename_result.second);
1124 
1126  }
1127 }
1128 
1129 int
1131 {
1132  int num_free = freeEntries[tid].robEntries -
1133  (instsInProgress[tid] - fromIEW->iewInfo[tid].dispatched);
1134 
1135  //DPRINTF(Rename,"[tid:%i] %i rob free\n",tid,num_free);
1136 
1137  return num_free;
1138 }
1139 
1140 int
1142 {
1143  int num_free = freeEntries[tid].iqEntries -
1144  (instsInProgress[tid] - fromIEW->iewInfo[tid].dispatched);
1145 
1146  //DPRINTF(Rename,"[tid:%i] %i iq free\n",tid,num_free);
1147 
1148  return num_free;
1149 }
1150 
1151 int
1153 {
1154  int num_free = freeEntries[tid].lqEntries -
1155  (loadsInProgress[tid] - fromIEW->iewInfo[tid].dispatchedToLQ);
1156  DPRINTF(Rename,
1157  "calcFreeLQEntries: free lqEntries: %d, loadsInProgress: %d, "
1158  "loads dispatchedToLQ: %d\n",
1159  freeEntries[tid].lqEntries, loadsInProgress[tid],
1160  fromIEW->iewInfo[tid].dispatchedToLQ);
1161  return num_free;
1162 }
1163 
1164 int
1166 {
1167  int num_free = freeEntries[tid].sqEntries -
1168  (storesInProgress[tid] - fromIEW->iewInfo[tid].dispatchedToSQ);
1169  DPRINTF(Rename, "calcFreeSQEntries: free sqEntries: %d, "
1170  "storesInProgress: %d, stores dispatchedToSQ: %d\n",
1171  freeEntries[tid].sqEntries, storesInProgress[tid],
1172  fromIEW->iewInfo[tid].dispatchedToSQ);
1173  return num_free;
1174 }
1175 
1176 unsigned
1178 {
1179  unsigned inst_count = 0;
1180 
1181  for (int i=0; i<fromDecode->size; i++) {
1182  if (!fromDecode->insts[i]->isSquashed())
1183  inst_count++;
1184  }
1185 
1186  return inst_count;
1187 }
1188 
1189 void
1191 {
1192  if (fromIEW->iewBlock[tid]) {
1193  stalls[tid].iew = true;
1194  }
1195 
1196  if (fromIEW->iewUnblock[tid]) {
1197  assert(stalls[tid].iew);
1198  stalls[tid].iew = false;
1199  }
1200 }
1201 
1202 bool
1204 {
1205  bool ret_val = false;
1206 
1207  if (stalls[tid].iew) {
1208  DPRINTF(Rename,"[tid:%i] Stall from IEW stage detected.\n", tid);
1209  ret_val = true;
1210  } else if (calcFreeROBEntries(tid) <= 0) {
1211  DPRINTF(Rename,"[tid:%i] Stall: ROB has 0 free entries.\n", tid);
1212  ret_val = true;
1213  } else if (calcFreeIQEntries(tid) <= 0) {
1214  DPRINTF(Rename,"[tid:%i] Stall: IQ has 0 free entries.\n", tid);
1215  ret_val = true;
1216  } else if (calcFreeLQEntries(tid) <= 0 && calcFreeSQEntries(tid) <= 0) {
1217  DPRINTF(Rename,"[tid:%i] Stall: LSQ has 0 free entries.\n", tid);
1218  ret_val = true;
1219  } else if (renameMap[tid]->numFreeEntries() <= 0) {
1220  DPRINTF(Rename,"[tid:%i] Stall: RenameMap has 0 free entries.\n", tid);
1221  ret_val = true;
1222  } else if (renameStatus[tid] == SerializeStall &&
1223  (!emptyROB[tid] || instsInProgress[tid])) {
1224  DPRINTF(Rename,"[tid:%i] Stall: Serialize stall and ROB is not "
1225  "empty.\n",
1226  tid);
1227  ret_val = true;
1228  }
1229 
1230  return ret_val;
1231 }
1232 
1233 void
1235 {
1236  if (fromIEW->iewInfo[tid].usedIQ)
1237  freeEntries[tid].iqEntries = fromIEW->iewInfo[tid].freeIQEntries;
1238 
1239  if (fromIEW->iewInfo[tid].usedLSQ) {
1240  freeEntries[tid].lqEntries = fromIEW->iewInfo[tid].freeLQEntries;
1241  freeEntries[tid].sqEntries = fromIEW->iewInfo[tid].freeSQEntries;
1242  }
1243 
1244  if (fromCommit->commitInfo[tid].usedROB) {
1245  freeEntries[tid].robEntries =
1246  fromCommit->commitInfo[tid].freeROBEntries;
1247  emptyROB[tid] = fromCommit->commitInfo[tid].emptyROB;
1248  }
1249 
1250  DPRINTF(Rename, "[tid:%i] Free IQ: %i, Free ROB: %i, "
1251  "Free LQ: %i, Free SQ: %i, FreeRM %i(%i %i %i %i %i %i)\n",
1252  tid,
1253  freeEntries[tid].iqEntries,
1254  freeEntries[tid].robEntries,
1255  freeEntries[tid].lqEntries,
1256  freeEntries[tid].sqEntries,
1257  renameMap[tid]->numFreeEntries(),
1258  renameMap[tid]->numFreeIntEntries(),
1259  renameMap[tid]->numFreeFloatEntries(),
1260  renameMap[tid]->numFreeVecEntries(),
1261  renameMap[tid]->numFreeVecElemEntries(),
1262  renameMap[tid]->numFreePredEntries(),
1263  renameMap[tid]->numFreeCCEntries());
1264 
1265  DPRINTF(Rename, "[tid:%i] %i instructions not yet in ROB\n",
1266  tid, instsInProgress[tid]);
1267 }
1268 
1269 bool
1271 {
1272  // Check if there's a squash signal, squash if there is
1273  // Check stall signals, block if necessary.
1274  // If status was blocked
1275  // check if stall conditions have passed
1276  // if so then go to unblocking
1277  // If status was Squashing
1278  // check if squashing is not high. Switch to running this cycle.
1279  // If status was serialize stall
1280  // check if ROB is empty and no insts are in flight to the ROB
1281 
1282  readFreeEntries(tid);
1283  readStallSignals(tid);
1284 
1285  if (fromCommit->commitInfo[tid].squash) {
1286  DPRINTF(Rename, "[tid:%i] Squashing instructions due to squash from "
1287  "commit.\n", tid);
1288 
1289  squash(fromCommit->commitInfo[tid].doneSeqNum, tid);
1290 
1291  return true;
1292  }
1293 
1294  if (checkStall(tid)) {
1295  return block(tid);
1296  }
1297 
1298  if (renameStatus[tid] == Blocked) {
1299  DPRINTF(Rename, "[tid:%i] Done blocking, switching to unblocking.\n",
1300  tid);
1301 
1302  renameStatus[tid] = Unblocking;
1303 
1304  unblock(tid);
1305 
1306  return true;
1307  }
1308 
1309  if (renameStatus[tid] == Squashing) {
1310  // Switch status to running if rename isn't being told to block or
1311  // squash this cycle.
1312  if (resumeSerialize) {
1313  DPRINTF(Rename,
1314  "[tid:%i] Done squashing, switching to serialize.\n", tid);
1315 
1317  return true;
1318  } else if (resumeUnblocking) {
1319  DPRINTF(Rename,
1320  "[tid:%i] Done squashing, switching to unblocking.\n",
1321  tid);
1322  renameStatus[tid] = Unblocking;
1323  return true;
1324  } else {
1325  DPRINTF(Rename, "[tid:%i] Done squashing, switching to running.\n",
1326  tid);
1327  renameStatus[tid] = Running;
1328  return false;
1329  }
1330  }
1331 
1332  if (renameStatus[tid] == SerializeStall) {
1333  // Stall ends once the ROB is free.
1334  DPRINTF(Rename, "[tid:%i] Done with serialize stall, switching to "
1335  "unblocking.\n", tid);
1336 
1337  DynInstPtr serial_inst = serializeInst[tid];
1338 
1339  renameStatus[tid] = Unblocking;
1340 
1341  unblock(tid);
1342 
1343  DPRINTF(Rename, "[tid:%i] Processing instruction [%lli] with "
1344  "PC %s.\n", tid, serial_inst->seqNum, serial_inst->pcState());
1345 
1346  // Put instruction into queue here.
1347  serial_inst->clearSerializeBefore();
1348 
1349  if (!skidBuffer[tid].empty()) {
1350  skidBuffer[tid].push_front(serial_inst);
1351  } else {
1352  insts[tid].push_front(serial_inst);
1353  }
1354 
1355  DPRINTF(Rename, "[tid:%i] Instruction must be processed by rename."
1356  " Adding to front of list.\n", tid);
1357 
1358  serializeInst[tid] = NULL;
1359 
1360  return true;
1361  }
1362 
1363  // If we've reached this point, we have not gotten any signals that
1364  // cause rename to change its status. Rename remains the same as before.
1365  return false;
1366 }
1367 
1368 void
1370 {
1371  if (inst_list.empty()) {
1372  // Mark a bit to say that I must serialize on the next instruction.
1373  serializeOnNextInst[tid] = true;
1374  return;
1375  }
1376 
1377  // Set the next instruction as serializing.
1378  inst_list.front()->setSerializeBefore();
1379 }
1380 
1381 void
1383 {
1384  switch (source) {
1385  case ROB:
1386  ++stats.ROBFullEvents;
1387  break;
1388  case IQ:
1389  ++stats.IQFullEvents;
1390  break;
1391  case LQ:
1392  ++stats.LQFullEvents;
1393  break;
1394  case SQ:
1395  ++stats.SQFullEvents;
1396  break;
1397  default:
1398  panic("Rename full stall stat should be incremented for a reason!");
1399  break;
1400  }
1401 }
1402 
1403 void
1405 {
1407 
1408  for (ThreadID tid = 0; tid < numThreads; tid++) {
1409 
1410  buf_it = historyBuffer[tid].begin();
1411 
1412  while (buf_it != historyBuffer[tid].end()) {
1413  cprintf("Seq num: %i\nArch reg[%s]: %i New phys reg:"
1414  " %i[%s] Old phys reg: %i[%s]\n",
1415  (*buf_it).instSeqNum,
1416  (*buf_it).archReg.className(),
1417  (*buf_it).archReg.index(),
1418  (*buf_it).newPhysReg->index(),
1419  (*buf_it).newPhysReg->className(),
1420  (*buf_it).prevPhysReg->index(),
1421  (*buf_it).prevPhysReg->className());
1422 
1423  buf_it++;
1424  }
1425  }
1426 }
1427 
1428 } // namespace o3
1429 } // namespace gem5
gem5::o3::Rename::isDrained
bool isDrained() const
Has the stage drained?
Definition: rename.cc:305
gem5::curTick
Tick curTick()
The universal simulation clock.
Definition: cur_tick.hh:46
fatal
#define fatal(...)
This implements a cprintf based fatal() function.
Definition: logging.hh:190
gem5::o3::Rename::decodeToRenameDelay
int decodeToRenameDelay
Delay between decode and rename, in ticks.
Definition: rename.hh:432
gem5::o3::Rename::freeEntries
FreeEntries freeEntries[MaxThreads]
Per-thread tracking of the number of free entries of back-end structures.
Definition: rename.hh:402
gem5::o3::Rename::storesInProgress
int storesInProgress[MaxThreads]
Count of Store instructions in progress that have been sent off to the IQ and ROB,...
Definition: rename.hh:381
gem5::o3::Rename::tick
void tick()
Ticks rename, which processes all input signals and attempts to rename as many instructions as possib...
Definition: rename.cc:389
gem5::o3::Rename::serializeOnNextInst
bool serializeOnNextInst[MaxThreads]
Records if rename needs to serialize on the next instruction for any thread.
Definition: rename.hh:426
warn
#define warn(...)
Definition: logging.hh:246
gem5::o3::Rename::drainSanityCheck
void drainSanityCheck() const
Perform sanity checks after a drain.
Definition: rename.cc:325
gem5::o3::Rename::squash
void squash(const InstSeqNum &squash_seq_num, ThreadID tid)
Squashes all instructions in a thread.
Definition: rename.cc:336
gem5::o3::Rename::iew_ptr
IEW * iew_ptr
Pointer to IEW stage.
Definition: rename.hh:157
gem5::cprintf
void cprintf(const char *format, const Args &...args)
Definition: cprintf.hh:155
gem5::o3::Rename::FreeEntries::sqEntries
unsigned sqEntries
Definition: rename.hh:396
gem5::RegId::getNumPinnedWrites
int getNumPinnedWrites() const
Definition: reg_class.hh:210
gem5::o3::Rename::dumpHistory
void dumpHistory()
Debugging function used to dump history buffer of renamings.
Definition: rename.cc:1404
gem5::o3::Rename::setRenameQueue
void setRenameQueue(TimeBuffer< RenameStruct > *rq_ptr)
Sets pointer to time buffer used to communicate to the next stage.
Definition: rename.cc:203
gem5::o3::Rename::calcFreeROBEntries
int calcFreeROBEntries(ThreadID tid)
Calculates the number of free ROB entries for a specific thread.
Definition: rename.cc:1130
gem5::o3::Rename::validInsts
unsigned validInsts()
Returns the number of valid instructions coming from decode.
Definition: rename.cc:1177
gem5::o3::Rename::RenameStats::blockCycles
statistics::Scalar blockCycles
Stat for total number of cycles spent blocking.
Definition: rename.hh:488
gem5::VecElemClass
@ VecElemClass
Vector Register Native Elem lane.
Definition: reg_class.hh:63
gem5::o3::Scoreboard::getReg
bool getReg(PhysRegIdPtr phys_reg) const
Checks if the register is ready.
Definition: scoreboard.hh:86
rename.hh
gem5::o3::Rename::setRenameMap
void setRenameMap(UnifiedRenameMap rm_ptr[MaxThreads])
Sets pointer to rename maps (per-thread structures).
Definition: rename.cc:286
gem5::o3::Rename::RenameStats::runCycles
statistics::Scalar runCycles
Stat for total number of cycles spent running normally.
Definition: rename.hh:493
gem5::o3::Rename::wroteToTimeBuffer
bool wroteToTimeBuffer
Variable that tracks if decode has written to the time buffer this cycle.
Definition: rename.hh:386
gem5::o3::Rename::_status
RenameStatus _status
Rename status.
Definition: rename.hh:111
gem5::o3::Rename::Squashing
@ Squashing
Definition: rename.hh:103
gem5::o3::Rename::resumeSerialize
bool resumeSerialize
Whether or not rename needs to resume a serialize instruction after squashing.
Definition: rename.hh:450
gem5::o3::Rename::instsInProgress
int instsInProgress[MaxThreads]
Count of instructions in progress that have been sent off to the IQ and ROB, but are not yet included...
Definition: rename.hh:371
gem5::o3::Rename::skidInsert
void skidInsert(ThreadID tid)
Inserts unused instructions from a given thread into the skid buffer, to be renamed once rename unblo...
Definition: rename.cc:762
gem5::o3::Rename::commitToRenameDelay
unsigned commitToRenameDelay
Delay between commit and rename, in ticks.
Definition: rename.hh:435
gem5::o3::Rename::removeFromHistory
void removeFromHistory(InstSeqNum inst_seq_num, ThreadID tid)
Removes a committed instruction's rename history.
Definition: rename.cc:962
gem5::o3::Rename::RenameStats::RenameStats
RenameStats(statistics::Group *parent)
Definition: rename.cc:97
gem5::o3::Rename::freeList
UnifiedFreeList * freeList
Free list interface.
Definition: rename.hh:360
gem5::CCRegClass
@ CCRegClass
Condition-code register.
Definition: reg_class.hh:65
gem5::o3::CPU::RenameIdx
@ RenameIdx
Definition: cpu.hh:509
gem5::o3::Rename::renameQueue
TimeBuffer< RenameStruct > * renameQueue
Rename instruction queue.
Definition: rename.hh:339
gem5::o3::Rename::setScoreboard
void setScoreboard(Scoreboard *_scoreboard)
Sets pointer to the scoreboard.
Definition: rename.cc:299
gem5::o3::Rename::RenameStats::renamedInsts
statistics::Scalar renamedInsts
Stat for total number of renamed instructions.
Definition: rename.hh:497
gem5::o3::Rename::RenameStats::vecPredLookups
statistics::Scalar vecPredLookups
Definition: rename.hh:523
gem5::o3::Rename::historyBuffer
std::list< RenameHistory > historyBuffer[MaxThreads]
A per-thread list of all destination register renames, used to either undo rename mappings or free ol...
Definition: rename.hh:321
gem5::o3::Rename::block
bool block(ThreadID tid)
Switches rename to blocking, and signals back that rename has become blocked.
Definition: rename.cc:864
gem5::o3::Rename::FreeEntries::iqEntries
unsigned iqEntries
Definition: rename.hh:393
gem5::o3::Rename::unblock
bool unblock(ThreadID tid)
Switches rename to unblocking if the skid buffer is empty, and signals back that rename has unblocked...
Definition: rename.cc:897
gem5::o3::Rename::timeBuffer
TimeBuffer< TimeStruct > * timeBuffer
Pointer to main time buffer used for backwards communication.
Definition: rename.hh:327
gem5::o3::Rename::renameDestRegs
void renameDestRegs(const DynInstPtr &inst, ThreadID tid)
Renames the destination registers of an instruction.
Definition: rename.cc:1077
gem5::o3::Rename::ppSquashInRename
ProbePointArg< SeqNumRegPair > * ppSquashInRename
To probe when an instruction is squashed and the register mapping for it needs to be undone.
Definition: rename.hh:124
gem5::o3::UnifiedFreeList
FreeList class that simply holds the list of free integer and floating point registers.
Definition: free_list.hh:122
gem5::o3::Rename::RenameStats::fpLookups
statistics::Scalar fpLookups
Definition: rename.hh:521
gem5::o3::Scoreboard
Implements a simple scoreboard to track which registers are ready.
Definition: scoreboard.hh:53
gem5::o3::Rename::numThreads
ThreadID numThreads
The number of threads active in rename.
Definition: rename.hh:457
gem5::o3::Rename::skidsEmpty
bool skidsEmpty()
Returns if all of the skid buffers are empty.
Definition: rename.cc:810
dyn_inst.hh
gem5::o3::Rename::RenameStats::SQFullEvents
statistics::Scalar SQFullEvents
Stat for total number of times that the SQ starts a stall in rename.
Definition: rename.hh:512
gem5::o3::Rename::sortInsts
void sortInsts()
Separates instructions from decode into individual lists of instructions sorted by thread.
Definition: rename.cc:795
gem5::ArmISA::i
Bitfield< 7 > i
Definition: misc_types.hh:67
gem5::o3::Rename::skidBufferMax
unsigned skidBufferMax
The maximum skid buffer size.
Definition: rename.hh:460
gem5::RegId::index
RegIndex index() const
Index accessors.
Definition: reg_class.hh:180
gem5::o3::Rename::RenameStats::undoneMaps
statistics::Scalar undoneMaps
Stat for total number of mappings that were undone due to a squash.
Definition: rename.hh:528
gem5::o3::Rename::Stalls::iew
bool iew
Definition: rename.hh:413
gem5::o3::Rename::RenameStats::serializeStallCycles
statistics::Scalar serializeStallCycles
Stat for total number of cycles spent stalling for a serializing inst.
Definition: rename.hh:491
gem5::o3::Rename::RenameStats::skidInsts
statistics::Scalar skidInsts
Number of instructions inserted into skid buffers.
Definition: rename.hh:534
gem5::o3::Rename::RenameStats::unblockCycles
statistics::Scalar unblockCycles
Stat for total number of cycles spent unblocking.
Definition: rename.hh:495
gem5::VecPredRegClass
@ VecPredRegClass
Definition: reg_class.hh:64
gem5::RefCountingPtr< DynInst >
gem5::TimeBuffer
Definition: timebuf.hh:40
gem5::o3::Rename::Blocked
@ Blocked
Definition: rename.hh:104
gem5::o3::Rename::Unblocking
@ Unblocking
Definition: rename.hh:105
gem5::o3::Rename::calcFreeLQEntries
int calcFreeLQEntries(ThreadID tid)
Calculates the number of free LQ entries for a specific thread.
Definition: rename.cc:1152
gem5::o3::Commit::numROBFreeEntries
size_t numROBFreeEntries(ThreadID tid)
Returns the number of free ROB entries for a specific thread.
Definition: commit.cc:493
gem5::o3::Rename::Inactive
@ Inactive
Definition: rename.hh:94
gem5::o3::Rename::RenameStats::serializing
statistics::Scalar serializing
Number of serialize instructions handled.
Definition: rename.hh:530
gem5::o3::CPU::deactivateStage
void deactivateStage(const StageIdx idx)
Changes a stage's status to inactive within the activity recorder.
Definition: cpu.hh:550
gem5::o3::Rename::decodeQueue
TimeBuffer< DecodeStruct > * decodeQueue
Decode instruction queue interface.
Definition: rename.hh:345
gem5::o3::Rename::rename
void rename(bool &status_change, ThreadID tid)
Determines what to do based on rename's current status.
Definition: rename.cc:452
gem5::o3::IEW::instQueue
InstructionQueue instQueue
Instruction queue.
Definition: iew.hh:355
gem5::o3::Rename::Running
@ Running
Definition: rename.hh:100
gem5::FloatRegClass
@ FloatRegClass
Floating-point register.
Definition: reg_class.hh:59
gem5::o3::Rename::Active
@ Active
Definition: rename.hh:93
gem5::o3::Rename::clearStates
void clearStates(ThreadID tid)
Clear all thread-specific states.
Definition: rename.cc:227
gem5::o3::Rename::renameSrcRegs
void renameSrcRegs(const DynInstPtr &inst, ThreadID tid)
Renames the source registers of an instruction.
Definition: rename.cc:1011
gem5::o3::Rename::ROB
@ ROB
Definition: rename.hh:467
gem5::o3::Rename::renameWidth
unsigned renameWidth
Rename width, in instructions.
Definition: rename.hh:438
gem5::o3::Rename::scoreboard
Scoreboard * scoreboard
Pointer to the scoreboard.
Definition: rename.hh:366
gem5::RegId::setNumPinnedWrites
void setNumPinnedWrites(int num_writes)
Definition: reg_class.hh:211
gem5::PhysRegId::flatIndex
const RegIndex & flatIndex() const
Flat index accessor.
Definition: reg_class.hh:289
gem5::o3::Rename::LQ
@ LQ
Definition: rename.hh:469
gem5::o3::CPU
O3CPU class, has each of the stages (fetch through commit) within it, as well as all of the time buff...
Definition: cpu.hh:94
gem5::o3::Rename::stalls
Stalls stalls[MaxThreads]
Tracks which stages are telling decode to stall.
Definition: rename.hh:418
gem5::ThreadContext
ThreadContext is the external interface to all thread state for anything outside of the CPU.
Definition: thread_context.hh:94
gem5::o3::Rename::IQ
@ IQ
Definition: rename.hh:468
gem5::ThreadContext::flattenRegId
virtual RegId flattenRegId(const RegId &reg_id) const =0
gem5::o3::Rename::name
std::string name() const
Returns the name of rename.
Definition: rename.cc:92
DPRINTF
#define DPRINTF(x,...)
Definition: trace.hh:186
ADD_STAT
#define ADD_STAT(n,...)
Convenience macro to add a stat to a statistics group.
Definition: group.hh:75
gem5::o3::Rename::updateStatus
void updateStatus()
Updates overall rename status based on all of the threads' statuses.
Definition: rename.cc:826
gem5::o3::Rename::Idle
@ Idle
Definition: rename.hh:101
gem5::o3::Rename::RenameStats::idleCycles
statistics::Scalar idleCycles
Stat for total number of cycles spent idle.
Definition: rename.hh:486
gem5::o3::Rename::fromDecode
TimeBuffer< DecodeStruct >::wire fromDecode
Wire to get decode's output from decode queue.
Definition: rename.hh:348
gem5::o3::Rename::ppRename
ProbePointArg< DynInstPtr > * ppRename
To probe when register renaming for an instruction is complete.
Definition: rename.hh:119
gem5::o3::Rename::RenameStats::tempSerializing
statistics::Scalar tempSerializing
Number of instructions marked as temporarily serializing.
Definition: rename.hh:532
gem5::o3::Rename::doSquash
void doSquash(const InstSeqNum &squash_seq_num, ThreadID tid)
Executes actual squash, removing squashed instructions.
Definition: rename.cc:917
gem5::o3::Rename::checkStall
bool checkStall(ThreadID tid)
Checks if any stages are telling rename to block.
Definition: rename.cc:1203
gem5::o3::Rename::resetStage
void resetStage()
Reset this pipeline stage.
Definition: rename.cc:248
gem5::o3::Rename::RenameHistory
Holds the information for each destination register rename.
Definition: rename.hh:297
gem5::o3::CPU::activateStage
void activateStage(const StageIdx idx)
Changes a stage's status to active within the activity recorder.
Definition: cpu.hh:543
gem5::o3::LSQ::numFreeLoadEntries
unsigned numFreeLoadEntries()
Returns the number of free load entries.
Definition: lsq.cc:505
gem5::o3::InstructionQueue::numFreeEntries
unsigned numFreeEntries()
Returns total number of free entries.
Definition: inst_queue.cc:509
gem5::o3::Rename::insts
InstQueue insts[MaxThreads]
Queue of all instructions coming from decode this cycle.
Definition: rename.hh:351
gem5::o3::Rename::calcFreeIQEntries
int calcFreeIQEntries(ThreadID tid)
Calculates the number of free IQ entries for a specific thread.
Definition: rename.cc:1141
gem5::o3::Rename::SQ
@ SQ
Definition: rename.hh:470
gem5::o3::MaxWidth
static constexpr int MaxWidth
Definition: limits.hh:37
gem5::o3::Rename::loadsInProgress
int loadsInProgress[MaxThreads]
Count of Load instructions in progress that have been sent off to the IQ and ROB, but are not yet inc...
Definition: rename.hh:376
gem5::RegId::className
const char * className() const
Return a const char* with the register class name.
Definition: reg_class.hh:208
gem5::RegId::classValue
RegClassType classValue() const
Class accessor.
Definition: reg_class.hh:206
gem5::o3::Rename::RenameStats::lookups
statistics::Scalar lookups
Stat for total number of source register rename lookups.
Definition: rename.hh:519
std::pair
STL pair class.
Definition: stl.hh:58
gem5::o3::Rename::cpu
CPU * cpu
Pointer to CPU.
Definition: rename.hh:324
gem5::o3::Rename::toIEWIndex
unsigned toIEWIndex
The index of the instruction in the time buffer to IEW that rename is currently using.
Definition: rename.hh:443
gem5::o3::Rename::renameMap
UnifiedRenameMap * renameMap[MaxThreads]
Rename map interface.
Definition: rename.hh:357
gem5::o3::Rename::RenameStats::fullRegistersEvents
statistics::Scalar fullRegistersEvents
Stat for total number of times that rename runs out of free registers to use to rename.
Definition: rename.hh:515
gem5::o3::Rename::RenameStats::renamedOperands
statistics::Scalar renamedOperands
Stat for total number of renamed destination registers.
Definition: rename.hh:517
gem5::o3::Scoreboard::unsetReg
void unsetReg(PhysRegIdPtr phys_reg)
Sets the register as not ready.
Definition: scoreboard.hh:123
gem5::o3::Rename::RenameStats::IQFullEvents
statistics::Scalar IQFullEvents
Stat for total number of times that the IQ starts a stall in rename.
Definition: rename.hh:506
gem5::o3::Rename::setTimeBuffer
void setTimeBuffer(TimeBuffer< TimeStruct > *tb_ptr)
Sets the main backwards communication time buffer pointer.
Definition: rename.cc:188
gem5::ProbePointArg
ProbePointArg generates a point for the class of Arg.
Definition: thermal_domain.hh:54
gem5::o3::Rename::serializeAfter
void serializeAfter(InstQueue &inst_list, ThreadID tid)
Either serializes on the next instruction available in the InstQueue, or records that it must seriali...
Definition: rename.cc:1369
gem5::o3::Rename::FreeEntries::robEntries
unsigned robEntries
Definition: rename.hh:394
gem5::IntRegClass
@ IntRegClass
Integer register.
Definition: reg_class.hh:58
gem5::o3::MaxThreads
static constexpr int MaxThreads
Definition: limits.hh:38
gem5::o3::Rename::Rename
Rename(CPU *_cpu, const O3CPUParams &params)
Rename constructor.
Definition: rename.cc:61
gem5::o3::Rename
Rename handles both single threaded and SMT rename.
Definition: rename.hh:78
gem5::o3::CPU::activityThisCycle
void activityThisCycle()
Records that there was time buffer activity this cycle.
Definition: cpu.hh:539
gem5::o3::Rename::iewToRenameDelay
int iewToRenameDelay
Delay between iew and rename, in ticks.
Definition: rename.hh:429
gem5::o3::Rename::stats
gem5::o3::Rename::RenameStats stats
gem5::o3::Rename::RenameStats::ROBFullEvents
statistics::Scalar ROBFullEvents
Stat for total number of times that the ROB starts a stall in rename.
Definition: rename.hh:503
gem5::o3::Rename::incrFullStat
void incrFullStat(const FullSource &source)
Function used to increment the stat that corresponds to the source of the stall.
Definition: rename.cc:1382
gem5::MiscRegClass
@ MiscRegClass
Control (misc) register.
Definition: reg_class.hh:66
gem5::statistics::DataWrap::prereq
Derived & prereq(const Stat &prereq)
Set the prerequisite stat and marks this stat to print at the end of simulation.
Definition: statistics.hh:372
gem5::PhysRegId::className
const char * className() const
Return a const char* with the register class name.
Definition: reg_class.hh:208
gem5::o3::Rename::calcFreeSQEntries
int calcFreeSQEntries(ThreadID tid)
Calculates the number of free SQ entries for a specific thread.
Definition: rename.cc:1165
gem5::o3::Rename::RenameStats::vecLookups
statistics::Scalar vecLookups
Definition: rename.hh:522
gem5::o3::Rename::setDecodeQueue
void setDecodeQueue(TimeBuffer< DecodeStruct > *dq_ptr)
Sets pointer to time buffer coming from decode.
Definition: rename.cc:212
gem5::o3::Rename::fromCommit
TimeBuffer< TimeStruct >::wire fromCommit
Wire to get commit's output from backwards time buffer.
Definition: rename.hh:333
std::deque< DynInstPtr >
gem5::o3::Rename::emptyROB
bool emptyROB[MaxThreads]
Records if the ROB is empty.
Definition: rename.hh:408
gem5::o3::Rename::SerializeStall
@ SerializeStall
Definition: rename.hh:106
gem5::o3::LSQ::numFreeStoreEntries
unsigned numFreeStoreEntries()
Returns the number of free store entries.
Definition: lsq.cc:522
reg_class.hh
gem5::o3::UnifiedRenameMap::lookup
PhysRegIdPtr lookup(const RegId &arch_reg) const
Look up the physical register mapped to an architectural register.
Definition: rename_map.hh:261
gem5::o3::Rename::checkSignalsAndUpdate
bool checkSignalsAndUpdate(ThreadID tid)
Checks the signals and updates the status.
Definition: rename.cc:1270
gem5::statistics::Group
Statistics container.
Definition: group.hh:93
gem5::InstSeqNum
uint64_t InstSeqNum
Definition: inst_seq.hh:40
gem5::PhysRegId
Physical register ID.
Definition: reg_class.hh:224
gem5::o3::Rename::startupStage
void startupStage()
Initializes variables for the stage.
Definition: rename.cc:221
gem5::o3::Rename::setFreeList
void setFreeList(UnifiedFreeList *fl_ptr)
Sets pointer to the free list.
Definition: rename.cc:293
gem5::o3::Rename::RenameStats::committedMaps
statistics::Scalar committedMaps
Stat for total number of committed renaming mappings.
Definition: rename.hh:525
gem5::o3::Rename::blockThisCycle
bool blockThisCycle
Whether or not rename needs to block this cycle.
Definition: rename.hh:446
gem5::o3::Rename::FullSource
FullSource
Enum to record the source of a structure full stall.
Definition: rename.hh:465
gem5::o3::Rename::fromIEW
TimeBuffer< TimeStruct >::wire fromIEW
Wire to get IEW's output from backwards time buffer.
Definition: rename.hh:330
gem5::o3::Rename::readStallSignals
void readStallSignals(ThreadID tid)
Reads signals telling rename to block/unblock.
Definition: rename.cc:1190
gem5::o3::Rename::regProbePoints
void regProbePoints()
Registers probes.
Definition: rename.cc:179
gem5::o3::Rename::serializeInst
DynInstPtr serializeInst[MaxThreads]
The serialize instruction that rename has stalled on.
Definition: rename.hh:421
gem5::statistics::DataWrap::flags
Derived & flags(Flags _flags)
Set the flags and marks this stat to print at the end of simulation.
Definition: statistics.hh:358
gem5::VecRegClass
@ VecRegClass
Vector Register.
Definition: reg_class.hh:61
gem5::o3::IEW::ldstQueue
LSQ ldstQueue
Load / store queue.
Definition: iew.hh:358
cpu.hh
std::list< ThreadID >
gem5::o3::Rename::renameInsts
void renameInsts(ThreadID tid)
Renames instructions for the given thread.
Definition: rename.cc:507
gem5::o3::Rename::RenameStats::squashedInsts
statistics::Scalar squashedInsts
Stat for total number of squashed instructions that rename discards.
Definition: rename.hh:500
gem5::o3::Rename::RenameStats::squashCycles
statistics::Scalar squashCycles
Stat for total number of cycles spent squashing.
Definition: rename.hh:484
gem5::o3::Rename::toDecode
TimeBuffer< TimeStruct >::wire toDecode
Wire to write infromation heading to previous stages.
Definition: rename.hh:336
gem5
Reference material can be found at the JEDEC website: UFS standard http://www.jedec....
Definition: tlb.cc:60
gem5::statistics::total
const FlagsType total
Print the total.
Definition: info.hh:60
limits.hh
gem5::o3::Rename::takeOverFrom
void takeOverFrom()
Takes over from another CPU's thread.
Definition: rename.cc:319
gem5::o3::Rename::activeThreads
std::list< ThreadID > * activeThreads
Pointer to the list of active threads.
Definition: rename.hh:363
gem5::o3::Rename::RenameStats::LQFullEvents
statistics::Scalar LQFullEvents
Stat for total number of times that the LQ starts a stall in rename.
Definition: rename.hh:509
gem5::o3::Rename::FreeEntries::lqEntries
unsigned lqEntries
Definition: rename.hh:395
gem5::o3::Rename::RenameStats::intLookups
statistics::Scalar intLookups
Definition: rename.hh:520
gem5::o3::UnifiedRenameMap::rename
RenameInfo rename(const RegId &arch_reg)
Tell rename map to get a new free physical register to remap the specified architectural register.
Definition: rename_map.hh:223
gem5::PhysRegId::index
RegIndex index() const
Visible RegId methods.
Definition: reg_class.hh:180
gem5::o3::Rename::toIEW
TimeBuffer< RenameStruct >::wire toIEW
Wire to write any information heading to IEW.
Definition: rename.hh:342
gem5::o3::Rename::renameStatus
ThreadStatus renameStatus[MaxThreads]
Per-thread status.
Definition: rename.hh:114
gem5::o3::Rename::resumeUnblocking
bool resumeUnblocking
Whether or not rename needs to resume clearing out the skidbuffer after squashing.
Definition: rename.hh:454
gem5::o3::Rename::readFreeEntries
void readFreeEntries(ThreadID tid)
Gets the number of free entries for a specific thread.
Definition: rename.cc:1234
gem5::o3::UnifiedFreeList::addReg
void addReg(PhysRegIdPtr freed_reg)
Adds a register back to the free list.
Definition: free_list.hh:303
gem5::ThreadID
int16_t ThreadID
Thread index/ID type.
Definition: types.hh:242
gem5::o3::Rename::commit_ptr
Commit * commit_ptr
Pointer to commit stage.
Definition: rename.hh:160
gem5::RegId
Register ID: describe an architectural register with its class and index.
Definition: reg_class.hh:113
gem5::o3::Rename::skidBuffer
InstQueue skidBuffer[MaxThreads]
Skid buffer between rename and decode.
Definition: rename.hh:354
panic
#define panic(...)
This implements a cprintf based panic() function.
Definition: logging.hh:178
gem5::o3::UnifiedRenameMap::setEntry
void setEntry(const RegId &arch_reg, PhysRegIdPtr phys_reg)
Update rename map with a specific mapping.
Definition: rename_map.hh:302
gem5::o3::Rename::setActiveThreads
void setActiveThreads(std::list< ThreadID > *at_ptr)
Sets pointer to list of active threads.
Definition: rename.cc:279
gem5::o3::UnifiedRenameMap
Unified register rename map for all classes of registers.
Definition: rename_map.hh:173

Generated on Wed May 4 2022 12:13:54 for gem5 by doxygen 1.8.17