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 MinorCPUParams ¶ms,
73 zeroReg(cpu.threads[0]->getIsaPtr()->regClasses().
75 issueLimit(params.executeIssueLimit),
76 memoryIssueLimit(params.executeMemoryIssueLimit),
77 commitLimit(params.executeCommitLimit),
78 memoryCommitLimit(params.executeMemoryCommitLimit),
79 processMoreThanOneInput(params.executeCycleInput),
80 fuDescriptions(*params.executeFuncUnits),
81 numFuncUnits(fuDescriptions.funcUnits.size()),
82 setTraceTimeOnCommit(params.executeSetTraceTimeOnCommit),
83 setTraceTimeOnIssue(params.executeSetTraceTimeOnIssue),
84 allowEarlyMemIssue(params.executeAllowEarlyMemoryIssue),
85 noCostFUIndex(fuDescriptions.funcUnits.size() + 1),
86 lsq(name_ +
".lsq", name_ +
".dcache_port",
88 params.executeMaxAccessesInMemory,
89 params.executeMemoryWidth,
90 params.executeLSQRequestsQueueSize,
91 params.executeLSQTransfersQueueSize,
92 params.executeLSQStoreBufferSize,
93 params.executeLSQMaxStoreBufferStoresPerCycle,
95 executeInfo(params.numThreads,
102 fatal(
"%s: executeCommitLimit must be >= 1 (%d)\n", name_,
107 fatal(
"%s: executeCommitLimit must be >= 1 (%d)\n", name_,
112 fatal(
"%s: executeMemoryIssueLimit must be >= 1 (%d)\n", name_,
117 fatal(
"%s: executeMemoryCommitLimit (%d) must be <="
118 " executeCommitLimit (%d)\n",
122 if (params.executeInputBufferSize < 1) {
123 fatal(
"%s: executeInputBufferSize must be >= 1 (%d)\n", name_,
124 params.executeInputBufferSize);
127 if (params.executeInputBufferSize < 1) {
128 fatal(
"%s: executeInputBufferSize must be >= 1 (%d)\n", name_,
129 params.executeInputBufferSize);
135 unsigned int total_slots = 0;
139 std::ostringstream fu_name;
145 total_slots += fu_description->
opLat;
147 fu_name << name_ <<
".fu." <<
i;
155 for (
int op_class = No_OpClass + 1; op_class <
Num_OpClasses; op_class++) {
156 bool found_fu =
false;
157 unsigned int fu_index = 0;
162 static_cast<OpClass
>(op_class)))
170 warn(
"No functional unit for OpClass %s\n",
171 enums::OpClassStrings[op_class]);
176 for (
ThreadID tid = 0; tid < params.numThreads; tid++) {
182 name_ +
".inputBuffer" + tid_str,
"insts",
183 params.executeInputBufferSize));
185 const auto ®Classes =
cpu.
threads[tid]->getIsaPtr()->regClasses();
188 scoreboard.emplace_back(name_ +
".scoreboard" + tid_str, regClasses);
193 name_ +
".inFlightInsts" + tid_str,
"insts", total_slots);
197 name_ +
".inFUMemInsts" + tid_str,
"insts", total_slots);
227 const std::unique_ptr<PCStateBase> pc_before(inst->pc->clone());
228 std::unique_ptr<PCStateBase> target(thread->
pcState().
clone());
234 inst->isLastOpInInst() &&
235 (inst->staticInst->isSerializeAfter() ||
236 inst->staticInst->isSquashAfter());
239 *pc_before, *target, (force_branch ?
" (forcing)" :
""));
242 bool must_branch = *pc_before != *target ||
250 inst->staticInst->advancePC(*target);
254 *pc_before, *target);
257 if (inst->predictedTaken && !force_branch) {
263 " none happened inst: %s\n",
264 inst->pc->instAddr(), inst->predictedTarget->instAddr(),
268 }
else if (*inst->predictedTarget == *target) {
273 DPRINTF(
Branch,
"Predicted a branch from 0x%x to 0x%x correctly"
275 inst->pc->instAddr(), inst->predictedTarget->instAddr(),
282 " but got the wrong target (actual: 0x%x) inst: %s\n",
283 inst->pc->instAddr(), inst->predictedTarget->instAddr(),
284 target->instAddr(), *inst);
288 }
else if (must_branch) {
290 DPRINTF(
Branch,
"Unpredicted branch from 0x%x to 0x%x inst: %s\n",
291 inst->pc->instAddr(), target->instAddr(), *inst);
320 (inst->isBubble() ?
executeInfo[tid].lastPredictionSeqNum
321 : inst->id.predictionSeqNum),
332 ThreadID thread_id = inst->id.threadId;
339 bool is_load = inst->staticInst->isLoad();
340 bool is_store = inst->staticInst->isStore();
341 bool is_atomic = inst->staticInst->isAtomic();
342 bool is_prefetch = inst->staticInst->isDataPrefetch();
346 bool use_context_predicate =
true;
348 if (inst->translationFault !=
NoFault) {
350 DPRINTF(MinorMem,
"Completing fault from DTLB access: %s\n",
351 inst->translationFault->name());
353 if (inst->staticInst->isPrefetch()) {
354 DPRINTF(MinorMem,
"Not taking fault on prefetch: %s\n",
355 inst->translationFault->name());
360 fault = inst->translationFault;
362 fault->invoke(thread, inst->staticInst);
364 }
else if (!packet) {
365 DPRINTF(MinorMem,
"Completing failed request inst: %s\n",
367 use_context_predicate =
false;
369 inst->staticInst->completeAcc(
nullptr, &context, inst->traceData);
370 }
else if (packet->
isError()) {
371 DPRINTF(MinorMem,
"Trying to commit error response: %s\n",
374 fatal(
"Received error response packet for inst: %s\n", *inst);
375 }
else if (is_store || is_load || is_prefetch || is_atomic) {
378 DPRINTF(MinorMem,
"Memory response inst: %s addr: 0x%x size: %d\n",
381 if (is_load && packet->
getSize() > 0) {
382 DPRINTF(MinorMem,
"Memory data[0]: 0x%x\n",
383 static_cast<unsigned int>(packet->
getConstPtr<uint8_t>()[0]));
387 fault = inst->staticInst->completeAcc(packet, &context,
392 DPRINTF(MinorMem,
"Fault in memory completeAcc: %s\n",
394 fault->invoke(thread, inst->staticInst);
402 fatal(
"There should only ever be reads, "
403 "writes or faults at this point\n");
408 if (inst->traceData) {
409 inst->traceData->setPredicate((use_context_predicate ?
422 return cpu.checkInterrupts(thread_id);
428 DPRINTF(MinorInterrupt,
"Considering interrupt status from PC: %s\n",
429 cpu.getContext(thread_id)->pcState());
431 Fault interrupt =
cpu.getInterruptController(thread_id)->getInterrupt();
435 cpu.getInterruptController(thread_id)->updateIntrInfo();
436 interrupt->invoke(
cpu.getContext(thread_id));
440 DPRINTF(MinorInterrupt,
"Invoking interrupt: %s to PC: %s\n",
441 interrupt->name(),
cpu.getContext(thread_id)->pcState());
455 bool &passed_predicate,
Fault &fault)
460 passed_predicate =
false;
468 std::unique_ptr<PCStateBase> old_pc(thread->
pcState().
clone());
473 DPRINTF(MinorExecute,
"Initiating memRef inst: %s\n", *inst);
475 Fault init_fault = inst->staticInst->initiateAcc(&context,
480 assert(inst->translationFault !=
NoFault);
486 inst->translationFault =
NoFault;
491 DPRINTF(MinorExecute,
"Fault on memory inst: %s"
492 " initiateAcc: %s\n", *inst, init_fault->name());
498 DPRINTF(MinorMem,
"No memory access for inst: %s\n", *inst);
505 inst->traceData->setPredicate(passed_predicate);
531 unsigned int ret =
index + 1;
533 if (ret == cycle_size)
546 ret = cycle_size - 1;
562 unsigned int fu_index = 0;
571 unsigned num_insts_issued = 0;
574 unsigned num_mem_insts_issued = 0;
578 unsigned num_insts_discarded = 0;
582 Fault fault = inst->fault;
583 bool discarded =
false;
584 bool issued_mem_ref =
false;
586 if (inst->isBubble()) {
589 }
else if (
cpu.getContext(thread_id)->status() ==
592 DPRINTF(MinorExecute,
"Discarding inst: %s from suspended"
597 }
else if (inst->id.streamSeqNum != thread.
streamSeqNum) {
598 DPRINTF(MinorExecute,
"Discarding inst: %s as its stream"
599 " state was unexpected, expected: %d\n",
616 DPRINTF(MinorExecute,
"Trying to issue inst: %s to FU: %d\n",
623 bool fu_is_capable = (!inst->isFault() ?
624 fu->provides(inst->staticInst->opClass()) :
true);
626 if (inst->isNoCostInst()) {
637 Cycles(0),
cpu.getContext(thread_id),
false);
641 inst->extraCommitDelay =
Cycles(0);
642 inst->extraCommitDelayExpr = NULL;
651 }
else if (!fu_is_capable ||
fu->alreadyPushed()) {
653 if (!fu_is_capable) {
654 DPRINTF(MinorExecute,
"Can't issue as FU: %d isn't"
655 " capable\n", fu_index);
657 DPRINTF(MinorExecute,
"Can't issue as FU: %d is"
658 " already busy\n", fu_index);
660 }
else if (
fu->stalled) {
661 DPRINTF(MinorExecute,
"Can't issue inst: %s into FU: %d,"
664 }
else if (!
fu->canInsert()) {
665 DPRINTF(MinorExecute,
"Can't issue inst: %s to busy FU"
666 " for another: %d cycles\n",
667 *inst,
fu->cyclesBeforeInsert());
670 fu->findTiming(inst->staticInst) : NULL);
677 &(
fu->cantForwardFromFUIndices);
680 DPRINTF(MinorExecute,
"Can't issue inst: %s as extra"
681 " decoding is suppressing it\n",
683 }
else if (!
scoreboard[thread_id].canInstIssue(inst,
684 src_latencies, cant_forward_from_fu_indices,
685 cpu.curCycle(),
cpu.getContext(thread_id)))
687 DPRINTF(MinorExecute,
"Can't issue inst: %s yet\n",
691 DPRINTF(MinorExecute,
"Issuing inst: %s"
692 " into FU %d\n", *inst,
696 TimingExpr *extra_dest_retire_lat_expr = NULL;
702 extra_dest_retire_lat =
704 extra_dest_retire_lat_expr =
710 issued_mem_ref = inst->isMemRef();
715 inst->fuIndex = fu_index;
716 inst->extraCommitDelay = extra_dest_retire_lat;
717 inst->extraCommitDelayExpr =
718 extra_dest_retire_lat_expr;
720 if (issued_mem_ref) {
725 inst->instToWaitFor =
726 scoreboard[thread_id].execSeqNumToWaitFor(inst,
727 cpu.getContext(thread_id));
732 DPRINTF(MinorExecute,
"A barrier will"
733 " cause a delay in mem ref issue of"
734 " inst: %s until after inst"
735 " %d(exec)\n", *inst,
738 inst->instToWaitFor =
741 DPRINTF(MinorExecute,
"Memory ref inst:"
742 " %s must wait for inst %d(exec)"
744 *inst, inst->instToWaitFor);
747 inst->canEarlyIssue =
true;
751 DPRINTF(MinorExecute,
"Pushing mem inst: %s\n",
765 fu->description.opLat +
766 extra_dest_retire_lat +
768 cpu.getContext(thread_id),
769 issued_mem_ref && extra_assumed_lat ==
Cycles(0));
783 DPRINTF(MinorExecute,
"Didn't issue inst: %s\n", *inst);
789 if (debug::MinorTrace && !inst->isBubble()) {
790 inst->minorTraceInst(*
this,
795 if (!discarded && inst->isInst() &&
796 inst->staticInst->isFullMemBarrier())
798 DPRINTF(MinorMem,
"Issuing memory barrier inst: %s\n", *inst);
803 inst->traceData->setWhen(
curTick());
807 num_mem_insts_issued++;
810 num_insts_discarded++;
811 }
else if (!inst->isBubble()) {
815 DPRINTF(MinorExecute,
"Reached inst issue limit\n");
819 DPRINTF(MinorExecute,
"Stepping to next inst inputIndex: %d\n",
831 DPRINTF(MinorExecute,
"Wrapping\n");
842 return num_insts_issued;
849 unsigned int num_pc_event_checks = 0;
855 cpu.
threads[thread_id]->pcEventQueue.service(oldPC, thread);
856 num_pc_event_checks++;
859 if (num_pc_event_checks > 1) {
864 return num_pc_event_checks > 1;
870 assert(!inst->isFault());
876 if (!inst->staticInst->isMicroop() || inst->staticInst->isLastMicroop())
889 [inst->staticInst->opClass()]++;
893 inst->traceData->setCPSeq(thread->
numOp);
895 cpu.probeInstCommit(inst->staticInst, inst->pc->instAddr());
901 bool &completed_mem_issue)
903 ThreadID thread_id = inst->id.threadId;
906 bool completed_inst =
true;
914 panic(
"We should never hit the case where we try to commit from a "
915 "suspended thread as the streamSeqNum should not match");
916 }
else if (inst->isFault()) {
920 DPRINTF(MinorExecute,
"Fault inst reached Execute: %s\n",
921 inst->fault->name());
924 inst->fault->invoke(thread, NULL);
927 }
else if (inst->staticInst->isMemRef()) {
940 bool predicate_passed =
false;
942 predicate_passed, fault);
944 if (completed_mem_inst && fault !=
NoFault) {
945 if (early_memory_issue) {
946 DPRINTF(MinorExecute,
"Fault in early executing inst: %s\n",
950 inst->canEarlyIssue =
false;
953 completed_inst =
false;
955 DPRINTF(MinorExecute,
"Fault in execute: %s\n",
957 fault->invoke(thread, NULL);
960 completed_inst =
true;
963 completed_inst = completed_mem_inst;
965 completed_mem_issue = completed_inst;
966 }
else if (inst->isInst() && inst->staticInst->isFullMemBarrier() &&
969 DPRINTF(MinorExecute,
"Can't commit data barrier inst: %s yet as"
970 " there isn't space in the store buffer\n", *inst);
972 completed_inst =
false;
973 }
else if (inst->isInst() && inst->staticInst->isQuiesce()
977 completed_inst =
false;
982 DPRINTF(MinorExecute,
"Committing inst: %s\n", *inst);
984 fault = inst->staticInst->execute(&context,
994 if (inst->traceData) {
995 if (debug::ExecFaulting) {
996 inst->traceData->setFaulting(
true);
998 delete inst->traceData;
999 inst->traceData = NULL;
1003 DPRINTF(MinorExecute,
"Fault in execute of inst: %s fault: %s\n",
1004 *inst, fault->name());
1005 fault->invoke(thread, inst->staticInst);
1012 if (completed_inst) {
1016 executeInfo[thread_id].lastPredictionSeqNum = inst->id.predictionSeqNum;
1019 if (!inst->isFault() &&
1025 auto &resume_pc =
cpu.getContext(thread_id)->pcState();
1027 assert(resume_pc.microPC() == 0);
1029 DPRINTF(MinorInterrupt,
"Suspending thread: %d from Execute"
1030 " inst: %s\n", thread_id, *inst);
1039 return completed_inst;
1077 bool completed_inst =
true;
1080 unsigned int num_insts_committed = 0;
1084 unsigned int num_mem_refs_committed = 0;
1086 if (only_commit_microops && !ex_info.
inFlightInsts->empty()) {
1087 DPRINTF(MinorInterrupt,
"Only commit microops %s %d\n",
1099 if (only_commit_microops) {
1100 DPRINTF(MinorInterrupt,
"Committing tail of insts before"
1108 head_inflight_inst->
inst->id.execSeqNum;
1115 bool committed_inst =
false;
1116 bool discard_inst =
false;
1117 bool completed_mem_ref =
false;
1118 bool issued_mem_ref =
false;
1119 bool early_memory_issue =
false;
1122 completed_inst =
false;
1134 DPRINTF(MinorExecute,
"Trying to commit canCommitInsts: %d\n",
1144 }
else if (mem_response &&
1148 discard_inst = inst->id.streamSeqNum !=
1151 DPRINTF(MinorExecute,
"Trying to commit mem response: %s\n",
1156 DPRINTF(MinorExecute,
"Discarding mem inst: %s as its"
1157 " stream state was unexpected, expected: %d\n",
1163 committed_inst =
true;
1166 completed_mem_ref =
true;
1167 completed_inst =
true;
1168 }
else if (can_commit_insts) {
1173 bool try_to_commit =
false;
1185 DPRINTF(MinorExecute,
"Trying to commit from mem FUs\n");
1194 if (!fu_inst->isBubble() &&
1196 fu_inst->canEarlyIssue &&
1198 head_exec_seq_num > fu_inst->instToWaitFor)
1200 DPRINTF(MinorExecute,
"Issuing mem ref early"
1201 " inst: %s instToWaitFor: %d\n",
1202 *(fu_inst), fu_inst->instToWaitFor);
1205 try_to_commit =
true;
1206 early_memory_issue =
true;
1207 completed_inst =
true;
1212 if (!completed_inst && inst->isNoCostInst()) {
1213 DPRINTF(MinorExecute,
"Committing no cost inst: %s", *inst);
1215 try_to_commit =
true;
1216 completed_inst =
true;
1221 if (!completed_inst && !inst->inLSQ) {
1222 DPRINTF(MinorExecute,
"Trying to commit from FUs\n");
1231 if (fu_inst.
inst->isBubble()) {
1233 completed_inst =
false;
1234 }
else if (fu_inst_seq_num != head_exec_seq_num) {
1240 }
else if (fu_inst.
inst->id == inst->id) {
1244 try_to_commit =
true;
1245 completed_inst =
true;
1249 if (try_to_commit) {
1250 discard_inst = inst->id.streamSeqNum !=
1255 if (!discard_inst) {
1261 if (inst->extraCommitDelayExpr) {
1262 DPRINTF(MinorExecute,
"Evaluating expression for"
1263 " extra commit delay inst: %s\n", *inst);
1270 uint64_t extra_delay = inst->extraCommitDelayExpr->
1273 DPRINTF(MinorExecute,
"Extra commit delay expr"
1274 " result: %d\n", extra_delay);
1276 if (extra_delay < 128) {
1277 inst->extraCommitDelay +=
Cycles(extra_delay);
1279 DPRINTF(MinorExecute,
"Extra commit delay was"
1280 " very long: %d\n", extra_delay);
1282 inst->extraCommitDelayExpr = NULL;
1287 if (inst->extraCommitDelay !=
Cycles(0)) {
1288 inst->minimumCommitCycle =
cpu.curCycle() +
1289 inst->extraCommitDelay;
1290 inst->extraCommitDelay =
Cycles(0);
1295 if (!inst->isFault() && inst->isMemRef() &&
1297 inst->id.execSeqNum &&
1300 DPRINTF(MinorExecute,
"Not committing inst: %s yet"
1301 " as there are incomplete barriers in flight\n",
1303 completed_inst =
false;
1304 }
else if (inst->minimumCommitCycle > now) {
1305 DPRINTF(MinorExecute,
"Not committing inst: %s yet"
1306 " as it wants to be stalled for %d more cycles\n",
1307 *inst, inst->minimumCommitCycle - now);
1308 completed_inst =
false;
1311 early_memory_issue, branch, fault,
1312 committed_inst, issued_mem_ref);
1316 completed_inst =
true;
1319 if (completed_inst) {
1325 DPRINTF(MinorExecute,
"Unstalling %d for inst %s\n", inst->fuIndex, inst->id);
1326 funcUnits[inst->fuIndex]->stalled =
false;
1331 DPRINTF(MinorExecute,
"No instructions to commit\n");
1332 completed_inst =
false;
1336 assert(!(discard_inst && !completed_inst));
1341 DPRINTF(MinorExecute,
"Discarding inst: %s as its stream"
1342 " state was unexpected, expected: %d\n",
1350 if (issued_mem_ref) {
1357 if (completed_inst && inst->isMemRef()) {
1367 if (completed_inst && !(issued_mem_ref && fault ==
NoFault)) {
1369 DPRINTF(MinorExecute,
"Completed inst: %s\n", *inst);
1373 inst->isLastOpInInst();
1384 if (inst->isInst() && inst->staticInst->isFullMemBarrier()) {
1385 DPRINTF(MinorMem,
"Completing memory barrier"
1386 " inst: %s committed: %d\n", *inst, committed_inst);
1390 scoreboard[thread_id].clearInstDests(inst, inst->isMemRef());
1394 if (committed_inst) {
1395 bool is_no_cost_inst = inst->isNoCostInst();
1399 if (debug::MinorTrace && !is_no_cost_inst)
1402 if (!is_no_cost_inst)
1403 num_insts_committed++;
1406 DPRINTF(MinorExecute,
"Reached inst commit limit\n");
1410 if (inst->traceData) {
1412 inst->traceData->setWhen(
curTick());
1413 inst->traceData->dump();
1416 if (completed_mem_ref)
1417 num_mem_refs_committed++;
1420 DPRINTF(MinorExecute,
"Reached mem ref commit limit\n");
1428 return executeInfo[thread_id].lastCommitWasEndOfMacroop &&
1435 if (!
inp.outputWire->isBubble())
1440 unsigned int num_issued = 0;
1447 bool interrupted =
false;
1456 DPRINTF(MinorInterrupt,
"Execute skipping a cycle to allow old"
1457 " branch to complete\n");
1464 DPRINTF(MinorExecute,
"Attempting to commit [tid:%d]\n",
1471 commit(commit_tid,
true,
false, branch);
1482 commit(commit_tid,
false,
true, branch);
1487 DPRINTF(MinorExecute,
"Committing micro-ops for interrupt[tid:%d]\n",
1489 bool only_commit_microops = interrupted &&
1491 commit(commit_tid, only_commit_microops,
false, branch);
1499 cpu.getContext(commit_tid)->pcState(), branch);
1509 DPRINTF(MinorExecute,
"Attempting to issue [tid:%d]\n",
1511 num_issued =
issue(issue_tid);
1519 bool can_issue_next =
false;
1521 for (
ThreadID tid = 0; tid <
cpu.numThreads; tid++) {
1525 unsigned int input_index =
executeInfo[tid].inputIndex;
1527 if (inst->isFault()) {
1528 can_issue_next =
true;
1529 }
else if (!inst->isBubble()) {
1530 next_issuable_insts.push_back(inst);
1535 bool becoming_stalled =
true;
1545 if (
fu->occupancy !=0 && !
fu->stalled)
1546 becoming_stalled =
false;
1552 for (
auto inst : next_issuable_insts) {
1553 if (!
fu->stalled &&
fu->provides(inst->staticInst->opClass()) &&
1554 scoreboard[inst->id.threadId].canInstIssue(inst,
1556 cpu.getContext(inst->id.threadId))) {
1557 can_issue_next =
true;
1563 bool head_inst_might_commit =
false;
1567 if (!info.inFlightInsts->empty()) {
1568 const QueuedInst &head_inst = info.inFlightInsts->front();
1570 if (head_inst.
inst->isNoCostInst()) {
1571 head_inst_might_commit =
true;
1575 fu->front().inst->id == head_inst.
inst->id) ||
1578 head_inst_might_commit =
true;
1585 DPRINTF(Activity,
"Need to tick num issued insts: %s%s%s%s%s%s\n",
1586 (num_issued != 0 ?
" (issued some insts)" :
""),
1587 (becoming_stalled ?
"(becoming stalled)" :
"(not becoming stalled)"),
1588 (can_issue_next ?
" (can issued next inst)" :
""),
1589 (head_inst_might_commit ?
"(head inst might commit)" :
""),
1591 (interrupted ?
" (interrupted)" :
""));
1595 !becoming_stalled ||
1597 head_inst_might_commit ||
1601 if (!need_to_tick) {
1602 DPRINTF(Activity,
"The next cycle might be skippable as there are no"
1603 " advanceable FUs\n");
1615 if (!
inp.outputWire->isBubble())
1627 bool thread_interrupted =
false;
1634 interrupted = interrupted || thread_interrupted;
1636 DPRINTF(MinorInterrupt,
"No interrupt controller\n");
1638 DPRINTF(MinorInterrupt,
"[tid:%d] thread_interrupted?=%d isInbetweenInsts?=%d\n",
1647 tid = (tid + 1) %
cpu.numThreads;
1668 std::ostringstream insts;
1669 std::ostringstream stalled;
1671 executeInfo[0].instsBeingCommitted.reportData(insts);
1680 stalled << (
funcUnits[
i]->stalled ?
'1' :
'E');
1687 " stalled=%s drainState=%d isInbetweenInsts=%d\n",
1704 case enums::SingleThreaded:
1706 case enums::RoundRobin:
1713 panic(
"Invalid thread policy");
1716 for (
auto tid : priority_list) {
1719 if (can_commit_insts) {
1723 can_commit_insts = can_commit_insts &&
1727 bool can_transfer_mem_inst =
false;
1733 can_transfer_mem_inst =
1734 !fu_inst->isBubble() &&
1735 fu_inst->id.threadId == tid &&
1737 fu_inst->canEarlyIssue &&
1738 inst->id.execSeqNum > fu_inst->instToWaitFor;
1742 if (can_commit_insts && !can_transfer_mem_inst &&
1746 can_execute_fu_inst = !fu_inst.
inst->isBubble() &&
1747 fu_inst.
inst->id == inst->id;
1750 can_commit_insts = can_commit_insts &&
1751 (can_transfer_mem_inst || can_execute_fu_inst);
1756 if (can_commit_insts) {
1771 case enums::SingleThreaded:
1773 case enums::RoundRobin:
1780 panic(
"Invalid thread scheduling policy.");
1783 for (
auto tid : priority_list) {
1796 DPRINTF(Drain,
"MinorExecute drainResume\n");
1798 for (
ThreadID tid = 0; tid <
cpu.numThreads; tid++) {
1810 os <<
"NotDraining";
1813 os <<
"DrainCurrentInst";
1816 os <<
"DrainHaltFetch";
1819 os <<
"DrainAllInsts";
1822 os <<
"Drain-" <<
static_cast<int>(state);
1832 DPRINTF(Drain,
"setDrainState[%d]: %s\n", thread_id, state);
1839 DPRINTF(Drain,
"MinorExecute drain\n");
1841 for (
ThreadID tid = 0; tid <
cpu.numThreads; tid++) {
1864 for (
ThreadID tid = 0; tid <
cpu.numThreads; tid++) {
1880 for (
ThreadID tid = 0; tid <
cpu.numThreads; tid++)
1887 return inst->id.streamSeqNum ==
executeInfo[inst->id.threadId].streamSeqNum;
1895 if (!
executeInfo[inst->id.threadId].inFlightInsts->empty())
1896 ret =
executeInfo[inst->id.threadId].inFlightInsts->front().inst->id == inst->id;