47 #include "debug/Activity.hh"
48 #include "debug/Branch.hh"
49 #include "debug/Drain.hh"
50 #include "debug/ExecFaulting.hh"
51 #include "debug/MinorExecute.hh"
52 #include "debug/MinorInterrupt.hh"
53 #include "debug/MinorMem.hh"
54 #include "debug/MinorTrace.hh"
55 #include "debug/PCEvent.hh"
66 const BaseMinorCPUParams ¶ms,
73 issueLimit(params.executeIssueLimit),
74 memoryIssueLimit(params.executeMemoryIssueLimit),
75 commitLimit(params.executeCommitLimit),
76 memoryCommitLimit(params.executeMemoryCommitLimit),
77 processMoreThanOneInput(params.executeCycleInput),
78 fuDescriptions(*params.executeFuncUnits),
79 numFuncUnits(fuDescriptions.funcUnits.size()),
80 setTraceTimeOnCommit(params.executeSetTraceTimeOnCommit),
81 setTraceTimeOnIssue(params.executeSetTraceTimeOnIssue),
82 allowEarlyMemIssue(params.executeAllowEarlyMemoryIssue),
83 noCostFUIndex(fuDescriptions.funcUnits.size() + 1),
84 lsq(name_ +
".lsq", name_ +
".dcache_port",
86 params.executeMaxAccessesInMemory,
87 params.executeMemoryWidth,
88 params.executeLSQRequestsQueueSize,
89 params.executeLSQTransfersQueueSize,
90 params.executeLSQStoreBufferSize,
91 params.executeLSQMaxStoreBufferStoresPerCycle),
92 executeInfo(params.numThreads,
99 fatal(
"%s: executeCommitLimit must be >= 1 (%d)\n", name_,
104 fatal(
"%s: executeCommitLimit must be >= 1 (%d)\n", name_,
109 fatal(
"%s: executeMemoryIssueLimit must be >= 1 (%d)\n", name_,
114 fatal(
"%s: executeMemoryCommitLimit (%d) must be <="
115 " executeCommitLimit (%d)\n",
119 if (params.executeInputBufferSize < 1) {
120 fatal(
"%s: executeInputBufferSize must be >= 1 (%d)\n", name_,
121 params.executeInputBufferSize);
124 if (params.executeInputBufferSize < 1) {
125 fatal(
"%s: executeInputBufferSize must be >= 1 (%d)\n", name_,
126 params.executeInputBufferSize);
132 unsigned int total_slots = 0;
136 std::ostringstream fu_name;
142 total_slots += fu_description->
opLat;
144 fu_name << name_ <<
".fu." <<
i;
152 for (
int op_class = No_OpClass + 1; op_class <
Num_OpClasses; op_class++) {
153 bool found_fu =
false;
154 unsigned int fu_index = 0;
159 static_cast<OpClass
>(op_class)))
167 warn(
"No functional unit for OpClass %s\n",
168 enums::OpClassStrings[op_class]);
173 for (
ThreadID tid = 0; tid < params.numThreads; tid++) {
179 name_ +
".inputBuffer" + tid_str,
"insts",
180 params.executeInputBufferSize));
182 const auto ®Classes =
cpu.
threads[tid]->getIsaPtr()->regClasses();
185 scoreboard.emplace_back(name_ +
".scoreboard" + tid_str, regClasses);
190 name_ +
".inFlightInsts" + tid_str,
"insts", total_slots);
194 name_ +
".inFUMemInsts" + tid_str,
"insts", total_slots);
224 const std::unique_ptr<PCStateBase> pc_before(inst->pc->clone());
225 std::unique_ptr<PCStateBase> target(thread->
pcState().
clone());
231 inst->isLastOpInInst() &&
232 (inst->staticInst->isSerializeAfter() ||
233 inst->staticInst->isSquashAfter());
236 *pc_before, *target, (force_branch ?
" (forcing)" :
""));
239 bool must_branch = *pc_before != *target ||
247 inst->staticInst->advancePC(*target);
251 *pc_before, *target);
254 if (inst->predictedTaken && !force_branch) {
260 " none happened inst: %s\n",
261 inst->pc->instAddr(), inst->predictedTarget->instAddr(),
265 }
else if (*inst->predictedTarget == *target) {
270 DPRINTF(
Branch,
"Predicted a branch from 0x%x to 0x%x correctly"
272 inst->pc->instAddr(), inst->predictedTarget->instAddr(),
279 " but got the wrong target (actual: 0x%x) inst: %s\n",
280 inst->pc->instAddr(), inst->predictedTarget->instAddr(),
281 target->instAddr(), *inst);
285 }
else if (must_branch) {
287 DPRINTF(
Branch,
"Unpredicted branch from 0x%x to 0x%x inst: %s\n",
288 inst->pc->instAddr(), target->instAddr(), *inst);
317 (inst->isBubble() ?
executeInfo[tid].lastPredictionSeqNum
318 : inst->id.predictionSeqNum),
329 ThreadID thread_id = inst->id.threadId;
336 bool is_load = inst->staticInst->isLoad();
337 bool is_store = inst->staticInst->isStore();
338 bool is_atomic = inst->staticInst->isAtomic();
339 bool is_prefetch = inst->staticInst->isDataPrefetch();
343 bool use_context_predicate =
true;
345 if (inst->translationFault !=
NoFault) {
347 DPRINTF(MinorMem,
"Completing fault from DTLB access: %s\n",
348 inst->translationFault->name());
350 if (inst->staticInst->isPrefetch()) {
351 DPRINTF(MinorMem,
"Not taking fault on prefetch: %s\n",
352 inst->translationFault->name());
357 fault = inst->translationFault;
359 fault->invoke(thread, inst->staticInst);
361 }
else if (!packet) {
362 DPRINTF(MinorMem,
"Completing failed request inst: %s\n",
364 use_context_predicate =
false;
366 inst->staticInst->completeAcc(
nullptr, &context, inst->traceData);
367 }
else if (packet->
isError()) {
368 DPRINTF(MinorMem,
"Trying to commit error response: %s\n",
371 fatal(
"Received error response packet for inst: %s\n", *inst);
372 }
else if (is_store || is_load || is_prefetch || is_atomic) {
375 DPRINTF(MinorMem,
"Memory response inst: %s addr: 0x%x size: %d\n",
378 if (is_load && packet->
getSize() > 0) {
379 DPRINTF(MinorMem,
"Memory data[0]: 0x%x\n",
380 static_cast<unsigned int>(packet->
getConstPtr<uint8_t>()[0]));
384 fault = inst->staticInst->completeAcc(packet, &context,
389 DPRINTF(MinorMem,
"Fault in memory completeAcc: %s\n",
391 fault->invoke(thread, inst->staticInst);
399 fatal(
"There should only ever be reads, "
400 "writes or faults at this point\n");
405 if (inst->traceData) {
406 inst->traceData->setPredicate((use_context_predicate ?
419 return cpu.checkInterrupts(thread_id);
425 DPRINTF(MinorInterrupt,
"Considering interrupt status from PC: %s\n",
426 cpu.getContext(thread_id)->pcState());
428 Fault interrupt =
cpu.getInterruptController(thread_id)->getInterrupt();
432 cpu.getInterruptController(thread_id)->updateIntrInfo();
433 interrupt->invoke(
cpu.getContext(thread_id));
437 DPRINTF(MinorInterrupt,
"Invoking interrupt: %s to PC: %s\n",
438 interrupt->name(),
cpu.getContext(thread_id)->pcState());
452 bool &passed_predicate,
Fault &fault)
457 passed_predicate =
false;
465 std::unique_ptr<PCStateBase> old_pc(thread->
pcState().
clone());
469 DPRINTF(MinorExecute,
"Initiating memRef inst: %s\n", *inst);
471 Fault init_fault = inst->staticInst->initiateAcc(&context,
476 assert(inst->translationFault !=
NoFault);
482 inst->translationFault =
NoFault;
487 DPRINTF(MinorExecute,
"Fault on memory inst: %s"
488 " initiateAcc: %s\n", *inst, init_fault->name());
494 DPRINTF(MinorMem,
"No memory access for inst: %s\n", *inst);
501 inst->traceData->setPredicate(passed_predicate);
527 unsigned int ret =
index + 1;
529 if (ret == cycle_size)
542 ret = cycle_size - 1;
558 unsigned int fu_index = 0;
567 unsigned num_insts_issued = 0;
570 unsigned num_mem_insts_issued = 0;
574 unsigned num_insts_discarded = 0;
578 Fault fault = inst->fault;
579 bool discarded =
false;
580 bool issued_mem_ref =
false;
582 if (inst->isBubble()) {
585 }
else if (
cpu.getContext(thread_id)->status() ==
588 DPRINTF(MinorExecute,
"Discarding inst: %s from suspended"
593 }
else if (inst->id.streamSeqNum != thread.
streamSeqNum) {
594 DPRINTF(MinorExecute,
"Discarding inst: %s as its stream"
595 " state was unexpected, expected: %d\n",
612 DPRINTF(MinorExecute,
"Trying to issue inst: %s to FU: %d\n",
619 bool fu_is_capable = (!inst->isFault() ?
620 fu->provides(inst->staticInst->opClass()) :
true);
622 if (inst->isNoCostInst()) {
633 Cycles(0),
cpu.getContext(thread_id),
false);
637 inst->extraCommitDelay =
Cycles(0);
638 inst->extraCommitDelayExpr = NULL;
647 }
else if (!fu_is_capable ||
fu->alreadyPushed()) {
649 if (!fu_is_capable) {
650 DPRINTF(MinorExecute,
"Can't issue as FU: %d isn't"
651 " capable\n", fu_index);
653 DPRINTF(MinorExecute,
"Can't issue as FU: %d is"
654 " already busy\n", fu_index);
656 }
else if (
fu->stalled) {
657 DPRINTF(MinorExecute,
"Can't issue inst: %s into FU: %d,"
660 }
else if (!
fu->canInsert()) {
661 DPRINTF(MinorExecute,
"Can't issue inst: %s to busy FU"
662 " for another: %d cycles\n",
663 *inst,
fu->cyclesBeforeInsert());
666 fu->findTiming(inst->staticInst) : NULL);
673 &(
fu->cantForwardFromFUIndices);
676 DPRINTF(MinorExecute,
"Can't issue inst: %s as extra"
677 " decoding is suppressing it\n",
679 }
else if (!
scoreboard[thread_id].canInstIssue(inst,
680 src_latencies, cant_forward_from_fu_indices,
681 cpu.curCycle(),
cpu.getContext(thread_id)))
683 DPRINTF(MinorExecute,
"Can't issue inst: %s yet\n",
687 DPRINTF(MinorExecute,
"Issuing inst: %s"
688 " into FU %d\n", *inst,
692 TimingExpr *extra_dest_retire_lat_expr = NULL;
698 extra_dest_retire_lat =
700 extra_dest_retire_lat_expr =
706 issued_mem_ref = inst->isMemRef();
711 inst->fuIndex = fu_index;
712 inst->extraCommitDelay = extra_dest_retire_lat;
713 inst->extraCommitDelayExpr =
714 extra_dest_retire_lat_expr;
716 if (issued_mem_ref) {
721 inst->instToWaitFor =
722 scoreboard[thread_id].execSeqNumToWaitFor(inst,
723 cpu.getContext(thread_id));
728 DPRINTF(MinorExecute,
"A barrier will"
729 " cause a delay in mem ref issue of"
730 " inst: %s until after inst"
731 " %d(exec)\n", *inst,
734 inst->instToWaitFor =
737 DPRINTF(MinorExecute,
"Memory ref inst:"
738 " %s must wait for inst %d(exec)"
740 *inst, inst->instToWaitFor);
743 inst->canEarlyIssue =
true;
747 DPRINTF(MinorExecute,
"Pushing mem inst: %s\n",
761 fu->description.opLat +
762 extra_dest_retire_lat +
764 cpu.getContext(thread_id),
765 issued_mem_ref && extra_assumed_lat ==
Cycles(0));
779 DPRINTF(MinorExecute,
"Didn't issue inst: %s\n", *inst);
785 if (debug::MinorTrace && !inst->isBubble()) {
786 inst->minorTraceInst(*
this,
791 if (!discarded && inst->isInst() &&
792 inst->staticInst->isFullMemBarrier())
794 DPRINTF(MinorMem,
"Issuing memory barrier inst: %s\n", *inst);
799 inst->traceData->setWhen(
curTick());
803 num_mem_insts_issued++;
806 num_insts_discarded++;
807 }
else if (!inst->isBubble()) {
811 DPRINTF(MinorExecute,
"Reached inst issue limit\n");
815 DPRINTF(MinorExecute,
"Stepping to next inst inputIndex: %d\n",
827 DPRINTF(MinorExecute,
"Wrapping\n");
838 return num_insts_issued;
845 unsigned int num_pc_event_checks = 0;
851 cpu.
threads[thread_id]->pcEventQueue.service(oldPC, thread);
852 num_pc_event_checks++;
855 if (num_pc_event_checks > 1) {
860 return num_pc_event_checks > 1;
866 assert(!inst->isFault());
872 if (!inst->staticInst->isMicroop() || inst->staticInst->isLastMicroop())
885 [inst->staticInst->opClass()]++;
889 inst->traceData->setCPSeq(thread->
numOp);
891 cpu.probeInstCommit(inst->staticInst, inst->pc->instAddr());
897 bool &completed_mem_issue)
899 ThreadID thread_id = inst->id.threadId;
902 bool completed_inst =
true;
910 panic(
"We should never hit the case where we try to commit from a "
911 "suspended thread as the streamSeqNum should not match");
912 }
else if (inst->isFault()) {
915 DPRINTF(MinorExecute,
"Fault inst reached Execute: %s\n",
916 inst->fault->name());
919 inst->fault->invoke(thread, NULL);
922 }
else if (inst->staticInst->isMemRef()) {
935 bool predicate_passed =
false;
937 predicate_passed, fault);
939 if (completed_mem_inst && fault !=
NoFault) {
940 if (early_memory_issue) {
941 DPRINTF(MinorExecute,
"Fault in early executing inst: %s\n",
945 inst->canEarlyIssue =
false;
948 completed_inst =
false;
950 DPRINTF(MinorExecute,
"Fault in execute: %s\n",
952 fault->invoke(thread, NULL);
955 completed_inst =
true;
958 completed_inst = completed_mem_inst;
960 completed_mem_issue = completed_inst;
961 }
else if (inst->isInst() && inst->staticInst->isFullMemBarrier() &&
964 DPRINTF(MinorExecute,
"Can't commit data barrier inst: %s yet as"
965 " there isn't space in the store buffer\n", *inst);
967 completed_inst =
false;
968 }
else if (inst->isInst() && inst->staticInst->isQuiesce()
972 completed_inst =
false;
976 DPRINTF(MinorExecute,
"Committing inst: %s\n", *inst);
978 fault = inst->staticInst->execute(&context,
988 if (inst->traceData) {
989 if (debug::ExecFaulting) {
990 inst->traceData->setFaulting(
true);
992 delete inst->traceData;
993 inst->traceData = NULL;
997 DPRINTF(MinorExecute,
"Fault in execute of inst: %s fault: %s\n",
998 *inst, fault->name());
999 fault->invoke(thread, inst->staticInst);
1006 if (completed_inst) {
1010 executeInfo[thread_id].lastPredictionSeqNum = inst->id.predictionSeqNum;
1013 if (!inst->isFault() &&
1019 auto &resume_pc =
cpu.getContext(thread_id)->pcState();
1021 assert(resume_pc.microPC() == 0);
1023 DPRINTF(MinorInterrupt,
"Suspending thread: %d from Execute"
1024 " inst: %s\n", thread_id, *inst);
1033 return completed_inst;
1071 bool completed_inst =
true;
1074 unsigned int num_insts_committed = 0;
1078 unsigned int num_mem_refs_committed = 0;
1080 if (only_commit_microops && !ex_info.
inFlightInsts->empty()) {
1081 DPRINTF(MinorInterrupt,
"Only commit microops %s %d\n",
1093 if (only_commit_microops) {
1094 DPRINTF(MinorInterrupt,
"Committing tail of insts before"
1102 head_inflight_inst->
inst->id.execSeqNum;
1109 bool committed_inst =
false;
1110 bool discard_inst =
false;
1111 bool completed_mem_ref =
false;
1112 bool issued_mem_ref =
false;
1113 bool early_memory_issue =
false;
1116 completed_inst =
false;
1128 DPRINTF(MinorExecute,
"Trying to commit canCommitInsts: %d\n",
1138 }
else if (mem_response &&
1142 discard_inst = inst->id.streamSeqNum !=
1145 DPRINTF(MinorExecute,
"Trying to commit mem response: %s\n",
1150 DPRINTF(MinorExecute,
"Discarding mem inst: %s as its"
1151 " stream state was unexpected, expected: %d\n",
1157 committed_inst =
true;
1160 completed_mem_ref =
true;
1161 completed_inst =
true;
1162 }
else if (can_commit_insts) {
1167 bool try_to_commit =
false;
1179 DPRINTF(MinorExecute,
"Trying to commit from mem FUs\n");
1188 if (!fu_inst->isBubble() &&
1190 fu_inst->canEarlyIssue &&
1192 head_exec_seq_num > fu_inst->instToWaitFor)
1194 DPRINTF(MinorExecute,
"Issuing mem ref early"
1195 " inst: %s instToWaitFor: %d\n",
1196 *(fu_inst), fu_inst->instToWaitFor);
1199 try_to_commit =
true;
1200 early_memory_issue =
true;
1201 completed_inst =
true;
1206 if (!completed_inst && inst->isNoCostInst()) {
1207 DPRINTF(MinorExecute,
"Committing no cost inst: %s", *inst);
1209 try_to_commit =
true;
1210 completed_inst =
true;
1215 if (!completed_inst && !inst->inLSQ) {
1216 DPRINTF(MinorExecute,
"Trying to commit from FUs\n");
1225 if (fu_inst.
inst->isBubble()) {
1227 completed_inst =
false;
1228 }
else if (fu_inst_seq_num != head_exec_seq_num) {
1234 }
else if (fu_inst.
inst->id == inst->id) {
1238 try_to_commit =
true;
1239 completed_inst =
true;
1243 if (try_to_commit) {
1244 discard_inst = inst->id.streamSeqNum !=
1249 if (!discard_inst) {
1255 if (inst->extraCommitDelayExpr) {
1256 DPRINTF(MinorExecute,
"Evaluating expression for"
1257 " extra commit delay inst: %s\n", *inst);
1264 uint64_t extra_delay = inst->extraCommitDelayExpr->
1267 DPRINTF(MinorExecute,
"Extra commit delay expr"
1268 " result: %d\n", extra_delay);
1270 if (extra_delay < 128) {
1271 inst->extraCommitDelay +=
Cycles(extra_delay);
1273 DPRINTF(MinorExecute,
"Extra commit delay was"
1274 " very long: %d\n", extra_delay);
1276 inst->extraCommitDelayExpr = NULL;
1281 if (inst->extraCommitDelay !=
Cycles(0)) {
1282 inst->minimumCommitCycle =
cpu.curCycle() +
1283 inst->extraCommitDelay;
1284 inst->extraCommitDelay =
Cycles(0);
1289 if (!inst->isFault() && inst->isMemRef() &&
1291 inst->id.execSeqNum &&
1294 DPRINTF(MinorExecute,
"Not committing inst: %s yet"
1295 " as there are incomplete barriers in flight\n",
1297 completed_inst =
false;
1298 }
else if (inst->minimumCommitCycle > now) {
1299 DPRINTF(MinorExecute,
"Not committing inst: %s yet"
1300 " as it wants to be stalled for %d more cycles\n",
1301 *inst, inst->minimumCommitCycle - now);
1302 completed_inst =
false;
1305 early_memory_issue, branch, fault,
1306 committed_inst, issued_mem_ref);
1310 completed_inst =
true;
1313 if (completed_inst) {
1319 DPRINTF(MinorExecute,
"Unstalling %d for inst %s\n", inst->fuIndex, inst->id);
1320 funcUnits[inst->fuIndex]->stalled =
false;
1325 DPRINTF(MinorExecute,
"No instructions to commit\n");
1326 completed_inst =
false;
1330 assert(!(discard_inst && !completed_inst));
1335 DPRINTF(MinorExecute,
"Discarding inst: %s as its stream"
1336 " state was unexpected, expected: %d\n",
1344 if (issued_mem_ref) {
1351 if (completed_inst && inst->isMemRef()) {
1361 if (completed_inst && !(issued_mem_ref && fault ==
NoFault)) {
1363 DPRINTF(MinorExecute,
"Completed inst: %s\n", *inst);
1367 inst->isLastOpInInst();
1378 if (inst->isInst() && inst->staticInst->isFullMemBarrier()) {
1379 DPRINTF(MinorMem,
"Completing memory barrier"
1380 " inst: %s committed: %d\n", *inst, committed_inst);
1384 scoreboard[thread_id].clearInstDests(inst, inst->isMemRef());
1388 if (committed_inst) {
1389 bool is_no_cost_inst = inst->isNoCostInst();
1393 if (debug::MinorTrace && !is_no_cost_inst)
1396 if (!is_no_cost_inst)
1397 num_insts_committed++;
1400 DPRINTF(MinorExecute,
"Reached inst commit limit\n");
1404 if (inst->traceData) {
1406 inst->traceData->setWhen(
curTick());
1407 inst->traceData->dump();
1410 if (completed_mem_ref)
1411 num_mem_refs_committed++;
1414 DPRINTF(MinorExecute,
"Reached mem ref commit limit\n");
1422 return executeInfo[thread_id].lastCommitWasEndOfMacroop &&
1429 if (!
inp.outputWire->isBubble())
1434 unsigned int num_issued = 0;
1441 bool interrupted =
false;
1450 DPRINTF(MinorInterrupt,
"Execute skipping a cycle to allow old"
1451 " branch to complete\n");
1458 DPRINTF(MinorExecute,
"Attempting to commit [tid:%d]\n",
1465 commit(commit_tid,
true,
false, branch);
1476 commit(commit_tid,
false,
true, branch);
1481 DPRINTF(MinorExecute,
"Committing micro-ops for interrupt[tid:%d]\n",
1483 bool only_commit_microops = interrupted &&
1485 commit(commit_tid, only_commit_microops,
false, branch);
1493 cpu.getContext(commit_tid)->pcState(), branch);
1503 DPRINTF(MinorExecute,
"Attempting to issue [tid:%d]\n",
1505 num_issued =
issue(issue_tid);
1513 bool can_issue_next =
false;
1515 for (
ThreadID tid = 0; tid <
cpu.numThreads; tid++) {
1519 unsigned int input_index =
executeInfo[tid].inputIndex;
1521 if (inst->isFault()) {
1522 can_issue_next =
true;
1523 }
else if (!inst->isBubble()) {
1524 next_issuable_insts.push_back(inst);
1529 bool becoming_stalled =
true;
1539 if (
fu->occupancy !=0 && !
fu->stalled)
1540 becoming_stalled =
false;
1546 for (
auto inst : next_issuable_insts) {
1547 if (!
fu->stalled &&
fu->provides(inst->staticInst->opClass()) &&
1548 scoreboard[inst->id.threadId].canInstIssue(inst,
1550 cpu.getContext(inst->id.threadId))) {
1551 can_issue_next =
true;
1557 bool head_inst_might_commit =
false;
1561 if (!info.inFlightInsts->empty()) {
1562 const QueuedInst &head_inst = info.inFlightInsts->front();
1564 if (head_inst.
inst->isNoCostInst()) {
1565 head_inst_might_commit =
true;
1569 fu->front().inst->id == head_inst.
inst->id) ||
1572 head_inst_might_commit =
true;
1579 DPRINTF(Activity,
"Need to tick num issued insts: %s%s%s%s%s%s\n",
1580 (num_issued != 0 ?
" (issued some insts)" :
""),
1581 (becoming_stalled ?
"(becoming stalled)" :
"(not becoming stalled)"),
1582 (can_issue_next ?
" (can issued next inst)" :
""),
1583 (head_inst_might_commit ?
"(head inst might commit)" :
""),
1585 (interrupted ?
" (interrupted)" :
""));
1589 !becoming_stalled ||
1591 head_inst_might_commit ||
1595 if (!need_to_tick) {
1596 DPRINTF(Activity,
"The next cycle might be skippable as there are no"
1597 " advanceable FUs\n");
1609 if (!
inp.outputWire->isBubble())
1621 bool thread_interrupted =
false;
1628 interrupted = interrupted || thread_interrupted;
1630 DPRINTF(MinorInterrupt,
"No interrupt controller\n");
1632 DPRINTF(MinorInterrupt,
"[tid:%d] thread_interrupted?=%d isInbetweenInsts?=%d\n",
1641 tid = (tid + 1) %
cpu.numThreads;
1662 std::ostringstream insts;
1663 std::ostringstream stalled;
1665 executeInfo[0].instsBeingCommitted.reportData(insts);
1674 stalled << (
funcUnits[
i]->stalled ?
'1' :
'E');
1681 " stalled=%s drainState=%d isInbetweenInsts=%d\n",
1698 case enums::SingleThreaded:
1700 case enums::RoundRobin:
1707 panic(
"Invalid thread policy");
1710 for (
auto tid : priority_list) {
1713 if (can_commit_insts) {
1717 can_commit_insts = can_commit_insts &&
1721 bool can_transfer_mem_inst =
false;
1727 can_transfer_mem_inst =
1728 !fu_inst->isBubble() &&
1729 fu_inst->id.threadId == tid &&
1731 fu_inst->canEarlyIssue &&
1732 inst->id.execSeqNum > fu_inst->instToWaitFor;
1736 if (can_commit_insts && !can_transfer_mem_inst &&
1740 can_execute_fu_inst = !fu_inst.
inst->isBubble() &&
1741 fu_inst.
inst->id == inst->id;
1744 can_commit_insts = can_commit_insts &&
1745 (can_transfer_mem_inst || can_execute_fu_inst);
1750 if (can_commit_insts) {
1765 case enums::SingleThreaded:
1767 case enums::RoundRobin:
1774 panic(
"Invalid thread scheduling policy.");
1777 for (
auto tid : priority_list) {
1790 DPRINTF(Drain,
"MinorExecute drainResume\n");
1792 for (
ThreadID tid = 0; tid <
cpu.numThreads; tid++) {
1804 os <<
"NotDraining";
1807 os <<
"DrainCurrentInst";
1810 os <<
"DrainHaltFetch";
1813 os <<
"DrainAllInsts";
1816 os <<
"Drain-" <<
static_cast<int>(
state);
1826 DPRINTF(Drain,
"setDrainState[%d]: %s\n", thread_id,
state);
1833 DPRINTF(Drain,
"MinorExecute drain\n");
1835 for (
ThreadID tid = 0; tid <
cpu.numThreads; tid++) {
1858 for (
ThreadID tid = 0; tid <
cpu.numThreads; tid++) {
1874 for (
ThreadID tid = 0; tid <
cpu.numThreads; tid++)
1881 return inst->id.streamSeqNum ==
executeInfo[inst->id.threadId].streamSeqNum;
1889 if (!
executeInfo[inst->id.threadId].inFlightInsts->empty())
1890 ret =
executeInfo[inst->id.threadId].inFlightInsts->front().inst->id == inst->id;