40 #include "arch/locked_mem.hh" 41 #include "arch/registers.hh" 42 #include "arch/utility.hh" 48 #include "debug/Activity.hh" 49 #include "debug/Branch.hh" 50 #include "debug/Drain.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" 62 MinorCPUParams ¶ms,
69 issueLimit(params.executeIssueLimit),
70 memoryIssueLimit(params.executeMemoryIssueLimit),
71 commitLimit(params.executeCommitLimit),
72 memoryCommitLimit(params.executeMemoryCommitLimit),
73 processMoreThanOneInput(params.executeCycleInput),
74 fuDescriptions(*params.executeFuncUnits),
75 numFuncUnits(fuDescriptions.funcUnits.size()),
76 setTraceTimeOnCommit(params.executeSetTraceTimeOnCommit),
77 setTraceTimeOnIssue(params.executeSetTraceTimeOnIssue),
78 allowEarlyMemIssue(params.executeAllowEarlyMemoryIssue),
79 noCostFUIndex(fuDescriptions.funcUnits.size() + 1),
80 lsq(name_ +
".lsq", name_ +
".dcache_port",
82 params.executeMaxAccessesInMemory,
83 params.executeMemoryWidth,
84 params.executeLSQRequestsQueueSize,
85 params.executeLSQTransfersQueueSize,
86 params.executeLSQStoreBufferSize,
87 params.executeLSQMaxStoreBufferStoresPerCycle),
94 fatal(
"%s: executeCommitLimit must be >= 1 (%d)\n", name_,
99 fatal(
"%s: executeCommitLimit must be >= 1 (%d)\n", name_,
104 fatal(
"%s: executeMemoryIssueLimit must be >= 1 (%d)\n", name_,
109 fatal(
"%s: executeMemoryCommitLimit (%d) must be <=" 110 " executeCommitLimit (%d)\n",
114 if (params.executeInputBufferSize < 1) {
115 fatal(
"%s: executeInputBufferSize must be >= 1 (%d)\n", name_,
116 params.executeInputBufferSize);
119 if (params.executeInputBufferSize < 1) {
120 fatal(
"%s: executeInputBufferSize must be >= 1 (%d)\n", name_,
121 params.executeInputBufferSize);
127 unsigned int total_slots = 0;
131 std::ostringstream fu_name;
137 total_slots += fu_description->
opLat;
139 fu_name << name_ <<
".fu." <<
i;
147 for (
int op_class = No_OpClass + 1; op_class <
Num_OpClasses; op_class++) {
148 bool found_fu =
false;
149 unsigned int fu_index = 0;
151 while (fu_index < numFuncUnits && !found_fu)
154 static_cast<OpClass>(op_class)))
162 warn(
"No functional unit for OpClass %s\n",
163 Enums::OpClassStrings[op_class]);
168 for (
ThreadID tid = 0; tid < params.numThreads; tid++) {
174 name_ +
".inputBuffer" + tid_str,
"insts",
175 params.executeInputBufferSize));
183 name_ +
".inFlightInsts" + tid_str,
"insts", total_slots);
187 name_ +
".inFUMemInsts" + tid_str,
"insts", total_slots);
224 inst->isLastOpInInst() &&
225 (inst->staticInst->isSerializeAfter() ||
226 inst->staticInst->isSquashAfter() ||
227 inst->staticInst->isIprAccess());
229 DPRINTF(Branch,
"tryToBranch before: %s after: %s%s\n",
230 pc_before, target, (force_branch ?
" (forcing)" :
""));
233 bool must_branch = pc_before != target ||
245 DPRINTF(Branch,
"Advancing current PC from: %s to: %s\n",
249 if (inst->predictedTaken && !force_branch) {
254 DPRINTF(Branch,
"Predicted a branch from 0x%x to 0x%x but" 255 " none happened inst: %s\n",
256 inst->pc.instAddr(), inst->predictedTarget.instAddr(), *inst);
259 }
else if (inst->predictedTarget == target) {
264 DPRINTF(Branch,
"Predicted a branch from 0x%x to 0x%x correctly" 266 inst->pc.instAddr(), inst->predictedTarget.instAddr(), *inst);
271 DPRINTF(Branch,
"Predicted a branch from 0x%x to 0x%x" 272 " but got the wrong target (actual: 0x%x) inst: %s\n",
273 inst->pc.instAddr(), inst->predictedTarget.instAddr(),
274 target.instAddr(), *inst);
278 }
else if (must_branch) {
280 DPRINTF(Branch,
"Unpredicted branch from 0x%x to 0x%x inst: %s\n",
281 inst->pc.instAddr(), target.instAddr(), *inst);
310 (inst->isBubble() ?
executeInfo[tid].lastPredictionSeqNum
311 : inst->id.predictionSeqNum),
314 DPRINTF(Branch,
"Branch data signalled: %s\n", branch);
322 ThreadID thread_id = inst->id.threadId;
329 bool is_load = inst->staticInst->isLoad();
330 bool is_store = inst->staticInst->isStore();
331 bool is_atomic = inst->staticInst->isAtomic();
332 bool is_prefetch = inst->staticInst->isDataPrefetch();
336 bool use_context_predicate =
true;
338 if (inst->translationFault !=
NoFault) {
340 DPRINTF(MinorMem,
"Completing fault from DTLB access: %s\n",
341 inst->translationFault->name());
343 if (inst->staticInst->isPrefetch()) {
344 DPRINTF(MinorMem,
"Not taking fault on prefetch: %s\n",
345 inst->translationFault->name());
350 fault = inst->translationFault;
352 fault->invoke(thread, inst->staticInst);
354 }
else if (!packet) {
355 DPRINTF(MinorMem,
"Completing failed request inst: %s\n",
357 use_context_predicate =
false;
359 inst->staticInst->completeAcc(
nullptr, &context, inst->traceData);
360 }
else if (packet->
isError()) {
361 DPRINTF(MinorMem,
"Trying to commit error response: %s\n",
364 fatal(
"Received error response packet for inst: %s\n", *inst);
365 }
else if (is_store || is_load || is_prefetch || is_atomic) {
368 DPRINTF(MinorMem,
"Memory response inst: %s addr: 0x%x size: %d\n",
371 if (is_load && packet->
getSize() > 0) {
372 DPRINTF(MinorMem,
"Memory data[0]: 0x%x\n",
373 static_cast<unsigned int>(packet->
getConstPtr<uint8_t>()[0]));
377 fault = inst->staticInst->completeAcc(packet, &context,
382 DPRINTF(MinorMem,
"Fault in memory completeAcc: %s\n",
384 fault->invoke(thread, inst->staticInst);
392 fatal(
"There should only ever be reads, " 393 "writes or faults at this point\n");
398 if (inst->traceData) {
399 inst->traceData->setPredicate((use_context_predicate ?
418 DPRINTF(MinorInterrupt,
"Considering interrupt status from PC: %s\n",
432 DPRINTF(MinorInterrupt,
"Invoking interrupt: %s to PC: %s\n",
447 bool &passed_predicate,
Fault &fault)
452 passed_predicate =
false;
465 DPRINTF(MinorExecute,
"Initiating memRef inst: %s\n", *inst);
467 Fault init_fault = inst->staticInst->initiateAcc(&context,
472 assert(inst->translationFault !=
NoFault);
478 inst->translationFault =
NoFault;
483 DPRINTF(MinorExecute,
"Fault on memory inst: %s" 484 " initiateAcc: %s\n", *inst, init_fault->name());
490 DPRINTF(MinorMem,
"No memory access for inst: %s\n", *inst);
497 inst->traceData->setPredicate(passed_predicate);
523 unsigned int ret = index + 1;
525 if (ret == cycle_size)
538 ret = cycle_size - 1;
554 unsigned int fu_index = 0;
563 unsigned num_insts_issued = 0;
566 unsigned num_mem_insts_issued = 0;
570 unsigned num_insts_discarded = 0;
574 Fault fault = inst->fault;
575 bool discarded =
false;
576 bool issued_mem_ref =
false;
578 if (inst->isBubble()) {
584 DPRINTF(MinorExecute,
"Discarding inst: %s from suspended" 589 }
else if (inst->id.streamSeqNum != thread.
streamSeqNum) {
590 DPRINTF(MinorExecute,
"Discarding inst: %s as its stream" 591 " state was unexpected, expected: %d\n",
608 DPRINTF(MinorExecute,
"Trying to issue inst: %s to FU: %d\n",
615 bool fu_is_capable = (!inst->isFault() ?
616 fu->
provides(inst->staticInst->opClass()) :
true);
618 if (inst->isNoCostInst()) {
633 inst->extraCommitDelay =
Cycles(0);
634 inst->extraCommitDelayExpr = NULL;
645 if (!fu_is_capable) {
646 DPRINTF(MinorExecute,
"Can't issue as FU: %d isn't" 647 " capable\n", fu_index);
649 DPRINTF(MinorExecute,
"Can't issue as FU: %d is" 650 " already busy\n", fu_index);
653 DPRINTF(MinorExecute,
"Can't issue inst: %s into FU: %d," 657 DPRINTF(MinorExecute,
"Can't issue inst: %s to busy FU" 658 " for another: %d cycles\n",
672 DPRINTF(MinorExecute,
"Can't issue inst: %s as extra" 673 " decoding is suppressing it\n",
675 }
else if (!
scoreboard[thread_id].canInstIssue(inst,
676 src_latencies, cant_forward_from_fu_indices,
679 DPRINTF(MinorExecute,
"Can't issue inst: %s yet\n",
683 DPRINTF(MinorExecute,
"Issuing inst: %s" 684 " into FU %d\n", *inst,
688 TimingExpr *extra_dest_retire_lat_expr = NULL;
694 extra_dest_retire_lat =
696 extra_dest_retire_lat_expr =
702 issued_mem_ref = inst->isMemRef();
707 inst->fuIndex = fu_index;
708 inst->extraCommitDelay = extra_dest_retire_lat;
709 inst->extraCommitDelayExpr =
710 extra_dest_retire_lat_expr;
712 if (issued_mem_ref) {
717 inst->instToWaitFor =
718 scoreboard[thread_id].execSeqNumToWaitFor(inst,
724 DPRINTF(MinorExecute,
"A barrier will" 725 " cause a delay in mem ref issue of" 726 " inst: %s until after inst" 727 " %d(exec)\n", *inst,
730 inst->instToWaitFor =
733 DPRINTF(MinorExecute,
"Memory ref inst:" 734 " %s must wait for inst %d(exec)" 736 *inst, inst->instToWaitFor);
739 inst->canEarlyIssue =
true;
743 DPRINTF(MinorExecute,
"Pushing mem inst: %s\n",
758 extra_dest_retire_lat +
761 issued_mem_ref && extra_assumed_lat ==
Cycles(0));
775 DPRINTF(MinorExecute,
"Didn't issue inst: %s\n", *inst);
781 if (
DTRACE(MinorTrace) && !inst->isBubble())
782 inst->minorTraceInst(*
this);
785 if (!discarded && inst->isInst() &&
786 inst->staticInst->isMemBarrier())
788 DPRINTF(MinorMem,
"Issuing memory barrier inst: %s\n", *inst);
793 inst->traceData->setWhen(
curTick());
797 num_mem_insts_issued++;
800 num_insts_discarded++;
801 }
else if (!inst->isBubble()) {
805 DPRINTF(MinorExecute,
"Reached inst issue limit\n");
809 DPRINTF(MinorExecute,
"Stepping to next inst inputIndex: %d\n",
821 DPRINTF(MinorExecute,
"Wrapping\n");
832 return num_insts_issued;
839 unsigned int num_pc_event_checks = 0;
845 cpu.
threads[thread_id]->pcEventQueue.service(oldPC, thread);
846 num_pc_event_checks++;
847 }
while (oldPC != thread->
instAddr());
849 if (num_pc_event_checks > 1) {
854 return num_pc_event_checks > 1;
860 assert(!inst->isFault());
866 if (!inst->staticInst->isMicroop() || inst->staticInst->isLastMicroop())
874 thread->comInstEventQueue.serviceEvents(thread->numInst);
880 [inst->staticInst->opClass()]++;
884 inst->traceData->setCPSeq(thread->numOp);
892 bool &completed_mem_issue)
894 ThreadID thread_id = inst->id.threadId;
897 bool completed_inst =
true;
905 panic(
"We should never hit the case where we try to commit from a " 906 "suspended thread as the streamSeqNum should not match");
907 }
else if (inst->isFault()) {
910 DPRINTF(MinorExecute,
"Fault inst reached Execute: %s\n",
911 inst->fault->name());
914 inst->fault->invoke(thread, NULL);
917 }
else if (inst->staticInst->isMemRef()) {
930 bool predicate_passed =
false;
932 predicate_passed, fault);
934 if (completed_mem_inst && fault !=
NoFault) {
935 if (early_memory_issue) {
936 DPRINTF(MinorExecute,
"Fault in early executing inst: %s\n",
940 inst->canEarlyIssue =
false;
943 completed_inst =
false;
945 DPRINTF(MinorExecute,
"Fault in execute: %s\n",
947 fault->invoke(thread, NULL);
950 completed_inst =
true;
953 completed_inst = completed_mem_inst;
955 completed_mem_issue = completed_inst;
956 }
else if (inst->isInst() && inst->staticInst->isMemBarrier() &&
959 DPRINTF(MinorExecute,
"Can't commit data barrier inst: %s yet as" 960 " there isn't space in the store buffer\n", *inst);
962 completed_inst =
false;
963 }
else if (inst->isInst() && inst->staticInst->isQuiesce()
967 completed_inst =
false;
971 DPRINTF(MinorExecute,
"Committing inst: %s\n", *inst);
973 fault = inst->staticInst->execute(&context,
983 DPRINTF(MinorExecute,
"Fault in execute of inst: %s fault: %s\n",
984 *inst, fault->name());
985 fault->invoke(thread, inst->staticInst);
992 if (completed_inst) {
996 executeInfo[thread_id].lastPredictionSeqNum = inst->id.predictionSeqNum;
999 if (!inst->isFault() &&
1007 assert(resume_pc.microPC() == 0);
1009 DPRINTF(MinorInterrupt,
"Suspending thread: %d from Execute" 1010 " inst: %s\n", thread_id, *inst);
1019 return completed_inst;
1057 bool completed_inst =
true;
1060 unsigned int num_insts_committed = 0;
1064 unsigned int num_mem_refs_committed = 0;
1066 if (only_commit_microops && !ex_info.
inFlightInsts->empty()) {
1067 DPRINTF(MinorInterrupt,
"Only commit microops %s %d\n",
1079 if (only_commit_microops) {
1080 DPRINTF(MinorInterrupt,
"Committing tail of insts before" 1088 head_inflight_inst->
inst->id.execSeqNum;
1095 bool committed_inst =
false;
1096 bool discard_inst =
false;
1097 bool completed_mem_ref =
false;
1098 bool issued_mem_ref =
false;
1099 bool early_memory_issue =
false;
1102 completed_inst =
false;
1114 DPRINTF(MinorExecute,
"Trying to commit canCommitInsts: %d\n",
1124 }
else if (mem_response &&
1128 discard_inst = inst->id.streamSeqNum !=
1131 DPRINTF(MinorExecute,
"Trying to commit mem response: %s\n",
1136 DPRINTF(MinorExecute,
"Discarding mem inst: %s as its" 1137 " stream state was unexpected, expected: %d\n",
1143 committed_inst =
true;
1146 completed_mem_ref =
true;
1147 completed_inst =
true;
1148 }
else if (can_commit_insts) {
1153 bool try_to_commit =
false;
1165 DPRINTF(MinorExecute,
"Trying to commit from mem FUs\n");
1174 if (!fu_inst->isBubble() &&
1176 fu_inst->canEarlyIssue &&
1178 head_exec_seq_num > fu_inst->instToWaitFor)
1180 DPRINTF(MinorExecute,
"Issuing mem ref early" 1181 " inst: %s instToWaitFor: %d\n",
1182 *(fu_inst), fu_inst->instToWaitFor);
1185 try_to_commit =
true;
1186 early_memory_issue =
true;
1187 completed_inst =
true;
1192 if (!completed_inst && inst->isNoCostInst()) {
1193 DPRINTF(MinorExecute,
"Committing no cost inst: %s", *inst);
1195 try_to_commit =
true;
1196 completed_inst =
true;
1201 if (!completed_inst && !inst->inLSQ) {
1202 DPRINTF(MinorExecute,
"Trying to commit from FUs\n");
1211 if (fu_inst.
inst->isBubble()) {
1213 completed_inst =
false;
1214 }
else if (fu_inst_seq_num != head_exec_seq_num) {
1220 }
else if (fu_inst.
inst->id == inst->id) {
1224 try_to_commit =
true;
1225 completed_inst =
true;
1229 if (try_to_commit) {
1230 discard_inst = inst->id.streamSeqNum !=
1235 if (!discard_inst) {
1241 if (inst->extraCommitDelayExpr) {
1242 DPRINTF(MinorExecute,
"Evaluating expression for" 1243 " extra commit delay inst: %s\n", *inst);
1250 uint64_t extra_delay = inst->extraCommitDelayExpr->
1253 DPRINTF(MinorExecute,
"Extra commit delay expr" 1254 " result: %d\n", extra_delay);
1256 if (extra_delay < 128) {
1257 inst->extraCommitDelay +=
Cycles(extra_delay);
1259 DPRINTF(MinorExecute,
"Extra commit delay was" 1260 " very long: %d\n", extra_delay);
1262 inst->extraCommitDelayExpr = NULL;
1267 if (inst->extraCommitDelay !=
Cycles(0)) {
1269 inst->extraCommitDelay;
1270 inst->extraCommitDelay =
Cycles(0);
1275 if (!inst->isFault() && inst->isMemRef() &&
1277 inst->id.execSeqNum &&
1280 DPRINTF(MinorExecute,
"Not committing inst: %s yet" 1281 " as there are incomplete barriers in flight\n",
1283 completed_inst =
false;
1284 }
else if (inst->minimumCommitCycle > now) {
1285 DPRINTF(MinorExecute,
"Not committing inst: %s yet" 1286 " as it wants to be stalled for %d more cycles\n",
1287 *inst, inst->minimumCommitCycle - now);
1288 completed_inst =
false;
1291 early_memory_issue, branch, fault,
1292 committed_inst, issued_mem_ref);
1296 completed_inst =
true;
1299 if (completed_inst) {
1305 DPRINTF(MinorExecute,
"Unstalling %d for inst %s\n", inst->fuIndex, inst->id);
1306 funcUnits[inst->fuIndex]->stalled =
false;
1311 DPRINTF(MinorExecute,
"No instructions to commit\n");
1312 completed_inst =
false;
1316 assert(!(discard_inst && !completed_inst));
1321 DPRINTF(MinorExecute,
"Discarding inst: %s as its stream" 1322 " state was unexpected, expected: %d\n",
1330 if (issued_mem_ref) {
1337 if (completed_inst && inst->isMemRef()) {
1347 if (completed_inst && !(issued_mem_ref && fault ==
NoFault)) {
1349 DPRINTF(MinorExecute,
"Completed inst: %s\n", *inst);
1353 inst->isLastOpInInst();
1364 if (inst->isInst() && inst->staticInst->isMemBarrier()) {
1365 DPRINTF(MinorMem,
"Completing memory barrier" 1366 " inst: %s committed: %d\n", *inst, committed_inst);
1370 scoreboard[thread_id].clearInstDests(inst, inst->isMemRef());
1374 if (committed_inst) {
1375 bool is_no_cost_inst = inst->isNoCostInst();
1379 if (
DTRACE(MinorTrace) && !is_no_cost_inst)
1382 if (!is_no_cost_inst)
1383 num_insts_committed++;
1386 DPRINTF(MinorExecute,
"Reached inst commit limit\n");
1390 if (inst->traceData) {
1392 inst->traceData->setWhen(
curTick());
1393 inst->traceData->dump();
1396 if (completed_mem_ref)
1397 num_mem_refs_committed++;
1400 DPRINTF(MinorExecute,
"Reached mem ref commit limit\n");
1408 return executeInfo[thread_id].lastCommitWasEndOfMacroop &&
1415 if (!
inp.outputWire->isBubble())
1420 unsigned int num_issued = 0;
1427 bool interrupted =
false;
1436 DPRINTF(MinorInterrupt,
"Execute skipping a cycle to allow old" 1437 " branch to complete\n");
1444 DPRINTF(MinorExecute,
"Attempting to commit [tid:%d]\n",
1451 commit(commit_tid,
true,
false, branch);
1462 commit(commit_tid,
false,
true, branch);
1467 DPRINTF(MinorExecute,
"Committing micro-ops for interrupt[tid:%d]\n",
1469 bool only_commit_microops = interrupted &&
1471 commit(commit_tid, only_commit_microops,
false, branch);
1488 DPRINTF(MinorExecute,
"Attempting to issue [tid:%d]\n",
1490 num_issued =
issue(issue_tid);
1498 bool can_issue_next =
false;
1504 unsigned int input_index =
executeInfo[tid].inputIndex;
1506 if (inst->isFault()) {
1507 can_issue_next =
true;
1508 }
else if (!inst->isBubble()) {
1509 next_issuable_insts.push_back(inst);
1514 bool becoming_stalled =
true;
1525 becoming_stalled =
false;
1531 for (
auto inst : next_issuable_insts) {
1533 scoreboard[inst->id.threadId].canInstIssue(inst,
1536 can_issue_next =
true;
1542 bool head_inst_might_commit =
false;
1546 if (!info.inFlightInsts->empty()) {
1547 const QueuedInst &head_inst = info.inFlightInsts->front();
1549 if (head_inst.
inst->isNoCostInst()) {
1550 head_inst_might_commit =
true;
1554 fu->
front().inst->id == head_inst.
inst->id) ||
1557 head_inst_might_commit =
true;
1564 DPRINTF(Activity,
"Need to tick num issued insts: %s%s%s%s%s%s\n",
1565 (num_issued != 0 ?
" (issued some insts)" :
""),
1566 (becoming_stalled ?
"(becoming stalled)" :
"(not becoming stalled)"),
1567 (can_issue_next ?
" (can issued next inst)" :
""),
1568 (head_inst_might_commit ?
"(head inst might commit)" :
""),
1570 (interrupted ?
" (interrupted)" :
""));
1574 !becoming_stalled ||
1576 head_inst_might_commit ||
1580 if (!need_to_tick) {
1581 DPRINTF(Activity,
"The next cycle might be skippable as there are no" 1582 " advanceable FUs\n");
1594 if (!
inp.outputWire->isBubble())
1606 bool thread_interrupted =
false;
1613 interrupted = interrupted || thread_interrupted;
1615 DPRINTF(MinorInterrupt,
"No interrupt controller\n");
1617 DPRINTF(MinorInterrupt,
"[tid:%d] thread_interrupted?=%d isInbetweenInsts?=%d\n",
1647 std::ostringstream insts;
1648 std::ostringstream stalled;
1650 executeInfo[0].instsBeingCommitted.reportData(insts);
1659 stalled << (
funcUnits[
i]->stalled ?
'1' :
'E');
1665 MINORTRACE(
"insts=%s inputIndex=%d streamSeqNum=%d" 1666 " stalled=%s drainState=%d isInbetweenInsts=%d\n",
1683 case Enums::SingleThreaded:
1685 case Enums::RoundRobin:
1692 panic(
"Invalid thread policy");
1695 for (
auto tid : priority_list) {
1698 if (can_commit_insts) {
1702 can_commit_insts = can_commit_insts &&
1706 bool can_transfer_mem_inst =
false;
1712 can_transfer_mem_inst =
1713 !fu_inst->isBubble() &&
1714 fu_inst->id.threadId == tid &&
1716 fu_inst->canEarlyIssue &&
1717 inst->id.execSeqNum > fu_inst->instToWaitFor;
1721 if (can_commit_insts && !can_transfer_mem_inst &&
1725 can_execute_fu_inst = !fu_inst.
inst->isBubble() &&
1726 fu_inst.
inst->id == inst->id;
1729 can_commit_insts = can_commit_insts &&
1730 (can_transfer_mem_inst || can_execute_fu_inst);
1735 if (can_commit_insts) {
1750 case Enums::SingleThreaded:
1752 case Enums::RoundRobin:
1759 panic(
"Invalid thread scheduling policy.");
1762 for (
auto tid : priority_list) {
1775 DPRINTF(Drain,
"MinorExecute drainResume\n");
1789 os <<
"NotDraining";
1792 os <<
"DrainCurrentInst";
1795 os <<
"DrainHaltFetch";
1798 os <<
"DrainAllInsts";
1801 os <<
"Drain-" <<
static_cast<int>(state);
1811 DPRINTF(Drain,
"setDrainState[%d]: %s\n", thread_id, state);
1818 DPRINTF(Drain,
"MinorExecute drain\n");
1866 return inst->id.streamSeqNum ==
executeInfo[inst->id.threadId].streamSeqNum;
1874 if (!
executeInfo[inst->id.threadId].inFlightInsts->empty())
1875 ret =
executeInfo[inst->id.threadId].inFlightInsts->front().inst->id == inst->id;
...ReportTraits are trait classes with the same functionality as ReportIF, but with elements explicit...
#define panic(...)
This implements a cprintf based panic() function.
DrainState
Stage cycle-by-cycle state.
void handleMemResponse(MinorDynInstPtr inst, LSQ::LSQRequestPtr response, BranchData &branch, Fault &fault)
Handle extracting mem ref responses from the memory queues and completing the associated instructions...
void popInput(ThreadID tid)
Pop an element off the input buffer, if there are any.
bool readMemAccPredicate() const override
Container class to box instructions in the FUs to make those queues have correct bubble behaviour whe...
void commit(ThreadID thread_id, bool only_commit_microops, bool discard, BranchData &branch)
Try and commit instructions from the ends of the functional unit pipelines.
virtual void probeInstCommit(const StaticInstPtr &inst, Addr pc)
Helper method to trigger PMU probes for a committed instruction.
decltype(nullptr) constexpr NoFault
Cycles is a wrapper class for representing cycle counts, i.e.
unsigned int cyclicIndexInc(unsigned int index, unsigned int cycle_size)
Increment a cyclic buffer index for indices [0, cycle_size-1].
#define fatal(...)
This implements a cprintf based fatal() function.
bool allowEarlyMemIssue
Allow mem refs to leave their FUs before reaching the head of the in flight insts queue if their depe...
static BranchData bubble()
BubbleIF interface.
ThreadID numThreads
Number of threads we're actually simulating (<= SMT_MAX_THREADS).
unsigned int memoryCommitLimit
Number of memory instructions that can be committed per cycle.
virtual TheISA::PCState pcState() const =0
static bool isStreamChange(const BranchData::Reason reason)
Is a request with this reason actually a request to change the PC rather than a bubble or branch pred...
void doInstCommitAccounting(MinorDynInstPtr inst)
Do the stats handling and instruction count and PC event events related to the new instruction/op cou...
void popResponse(LSQRequestPtr response)
Sanity check and pop the head response.
unsigned int cyclicIndexDec(unsigned int index, unsigned int cycle_size)
Decrement a cyclic buffer index for indices [0, cycle_size-1].
void issuedMemBarrierInst(MinorDynInstPtr inst)
A memory barrier instruction has been issued, remember its execSeqNum that we can avoid issuing memor...
bool instIsHeadInst(MinorDynInstPtr inst)
Returns true if the given instruction is at the head of the inFlightInsts instruction queue...
MinorCPU & cpu
Pointer back to the containing CPU.
void pushFailedRequest(MinorDynInstPtr inst)
Push a predicate failed-representing request into the queues just to maintain commit order...
All the fun of executing instructions from Decode and sending branch/new instruction stream info...
void wakeupOnEvent(unsigned int stage_id)
Interface for stages to signal that they have become active after a callback or eventq event where th...
void setDrainState(ThreadID thread_id, DrainState state)
Set the drain state (with useful debugging messages)
bool setTraceTimeOnCommit
Modify instruction trace times on commit.
unsigned int memoryIssueLimit
Number of memory ops that can be issued per cycle.
void tryToBranch(MinorDynInstPtr inst, Fault fault, BranchData &branch)
Generate Branch data based (into branch) on an observed (or not) change in PC while executing an inst...
unsigned int width() const
Number of instructions carried by this object.
unsigned int issue(ThreadID thread_id)
Try and issue instructions from the inputBuffer.
void activity()
Records that there is activity this cycle.
bool FullSystem
The FullSystem variable can be used to determine the current mode of simulation.
Object to gather the visible context for evaluation.
unsigned int commitLimit
Number of instructions that can be committed per cycle.
ThreadID getCommittingThread()
Use the current threading policy to determine the next thread to decode from.
bool needsToBeSentToStoreBuffer()
This request, once processed by the requests/transfers queues, will need to go to the store buffer...
The SimpleThread object provides a combination of the ThreadState object and the ThreadContext interf...
ElemType & front()
Peek at the end element of the pipe.
Minor contains all the definitions within the MinorCPU apart from the CPU class itself.
Queue< QueuedInst, ReportTraitsAdaptor< QueuedInst > > * inFlightInsts
In-order instructions either in FUs or the LSQ.
A functional unit that can execute any of opClasses operations with a single op(eration)Lat(ency) and...
std::vector< FUPipeline * > funcUnits
The execution functional units.
LSQRequestPtr findResponse(MinorDynInstPtr inst)
Returns a response if it's at the head of the transfers queue and it's either complete or can be sent...
Stats::Scalar numOps
Number of simulated insts and microops.
Cycles opLat
Delay from issuing the operation, to it reaching the end of the associated pipeline.
Wrapper for a queue type to act as a pipeline stage input queue.
unsigned int numFuncUnits
Number of functional units to produce.
Cycles extraAssumedLat
Extra delay that results should show in the scoreboard after leaving the pipeline.
ThreadContext is the external interface to all thread state for anything outside of the CPU...
virtual Fault getInterrupt(ThreadContext *tc)=0
ExecContext bears the exec_context interface for Minor.
const MinorFU & description
Functional unit description that this pipeline implements.
ExecContext bears the exec_context interface for Minor.
InstSeqNum lastPredictionSeqNum
A prediction number for use where one isn't available from an instruction.
Derived SenderState to carry data access info.
InstSeqNum streamSeqNum
Source of sequence number for instuction streams.
Stats::Scalar numFetchSuspends
Number of times fetch was asked to suspend by Execute.
Provide a non-protected base class for Minor's Ports as derived classes are created by Fetch1 and Exe...
Execute(const std::string &name_, MinorCPU &cpu_, MinorCPUParams ¶ms, Latch< ForwardInstData >::Output inp_, Latch< BranchData >::Input out_)
ForwardInstData instsBeingCommitted
Structure for reporting insts currently being processed/retired for MinorTrace.
virtual void updateIntrInfo(ThreadContext *tc)=0
bool canRequest()
Is their space in the request queue to be able to push a request by issuing an isMemRef instruction...
std::vector< MinorFU * > funcUnits
void completeMemBarrierInst(MinorDynInstPtr inst, bool committed)
Complete a barrier instruction.
bool provides(OpClass capability)
ThreadID getIssuingThread()
Tick curTick()
The current simulated tick.
bool takeInterrupt(ThreadID thread_id, BranchData &branch)
Act on an interrupt.
unsigned int noCostFUIndex
The FU index of the non-existent costless FU for instructions which pass the MinorDynInst::isNoCostIn...
std::vector< InputBuffer< ForwardInstData > > inputBuffer
Stats::Scalar numInsts
Number of simulated instructions.
std::vector< ThreadID > randomPriority()
bool checkInterrupts(ThreadContext *tc) const
bool isBubble() const
BubbleIF interface.
MinorDynInstPtr insts[MAX_FORWARD_INSTS]
Array of carried insts, ref counted.
bool tryPCEvents(ThreadID thread_id)
Try to act on PC-related events.
MinorCPU::MinorCPUPort & getDcachePort()
Return the raw-bindable port.
Minor::MinorActivityRecorder * activityRecorder
Activity recording for pipeline.
Cycles curCycle() const
Determine the current cycle, corresponding to a tick aligned to a clock edge.
bool canPushIntoStoreBuffer() const
Must check this before trying to insert into the store buffer.
void advancePC(PCState &pc, const StaticInstPtr &inst)
unsigned int inputIndex
Index that we've completed upto in getInput data.
std::vector< Cycles > srcRegsRelativeLats
Cycle offsets from the scoreboard delivery times of register values for each of this instruction's so...
virtual Addr instAddr() const =0
Extra timing capability to allow individual ops to have their source register dependency latencies tw...
bool isInterrupted(ThreadID thread_id) const
Has an interrupt been raised.
unsigned int occupancy
The number of slots with non-bubbles in them.
Latch< BranchData >::Input out
Input port carrying stream changes to Fetch1.
uint64_t Addr
Address type This will probably be moved somewhere else in the near future.
MinorFUPool & fuDescriptions
Descriptions of the functional units we want to generate.
bool isDrained()
Is there nothing left in the LSQ.
A Packet is used to encapsulate a transfer between two objects in the memory system (e...
MinorFUTiming * findTiming(const StaticInstPtr &inst)
Find the extra timing information for this instruction.
LSQ lsq
Dcache port to pass on to the CPU.
std::vector< ThreadID > roundRobinPriority(ThreadID priority)
Thread scheduling utility functions.
const ThreadID InvalidThreadID
void step()
Step checks the queues to see if their are issuable transfers which were not otherwise picked up by t...
Fetch1 is responsible for fetching "lines" from memory and passing them to Fetch2.
bool hasInterrupt(ThreadID thread_id)
Checks if a specific thread has an interrupt.
A load/store queue that allows outstanding reads and writes.
bool processMoreThanOneInput
If true, more than one input line can be processed each cycle if there is room to execute more instru...
A scoreboard of register dependencies including, for each register: The number of in-flight instructi...
Stats::Scalar numDiscardedOps
Number of ops discarded before committing.
bool readPredicate() const override
bool isInbetweenInsts(ThreadID thread_id) const
Are we between instructions? Can we be interrupted?
ThreadID checkInterrupts(BranchData &branch, bool &interrupted)
Check all threads for possible interrupts.
bool alreadyPushed()
Have we already pushed onto this pipe without advancing.
int16_t ThreadID
Thread index/ID type.
DrainState drainState
State progression for draining NotDraining -> ...
void sendStoreToStoreBuffer(LSQRequestPtr request)
A store has been committed, please move it to the store buffer.
bool suppress
If true, instructions matching this mask/match should not be issued in this FU.
TimingExpr * extraCommitLatExpr
Forward data betwen Execute and Fetch1 carrying change-of-address/stream information.
ThreadID interruptPriority
Cycles cyclesBeforeInsert()
How many cycles must from curCycle before insertion into the pipeline is allowed. ...
bool lastCommitWasEndOfMacroop
The last commit was the end of a full instruction so an interrupt can safely happen.
Enums::ThreadPolicy threadPolicy
Thread Scheduling Policy (RoundRobin, Random, etc)
bool instIsRightStream(MinorDynInstPtr inst)
Does the given instruction have the right stream sequence number to be committed? ...
void advance()
Step the pipeline.
static MinorDynInstPtr bubble()
There is a single bubble inst.
static const OpClass Num_OpClasses
bool canInsert() const
Can an instruction be inserted now?
Top level definition of the Minor in-order CPU model.
bool executeMemRefInst(MinorDynInstPtr inst, BranchData &branch, bool &failed_predicate, Fault &fault)
Execute a memory reference instruction.
Stats::Vector2d committedInstType
Number of instructions by type (OpClass)
virtual ThreadContext * getContext(int tn)
Given a thread num get tho thread context for it.
std::vector< Minor::MinorThread * > threads
These are thread state-representing objects for this CPU.
bool commitInst(MinorDynInstPtr inst, bool early_memory_issue, BranchData &branch, Fault &fault, bool &committed, bool &completed_mem_issue)
Commit a single instruction.
std::vector< ExecuteThreadInfo > executeInfo
#define MINORTRACE(...)
DPRINTFN for MinorTrace reporting.
BaseInterrupts * getInterruptController(ThreadID tid)
std::vector< Scoreboard > scoreboard
Scoreboard of instruction dependencies.
const T * getConstPtr() const
virtual Status status() const =0
void minorTrace() const
Report buffer states from 'slot' 'from' to 'to'.
void push(ElemType &elem)
Write an element to the back of the pipeline.
InstSeqNum getLastMemBarrier(ThreadID thread_id) const
Get the execSeqNum of the last issued memory barrier.
MinorCPU is an in-order CPU model with four fixed pipeline stages:
bool isDrained()
After thread suspension, has Execute been drained of in-flight instructions and memory accesses...
Latch< ForwardInstData >::Output inp
Input port carrying instructions from Decode.
Cycles extraCommitLat
Extra latency that the instruction should spend at the end of the pipeline.
Minor::MinorStats stats
Processor-specific statistics.
GenericISA::DelaySlotPCState< MachInst > PCState
void evaluate()
Pass on input/buffer data to the output if you can.
bool stalled
If true, advance will not advance the pipeline.
unsigned int drain()
Like the drain interface on SimObject.
std::shared_ptr< FaultBase > Fault
MinorCPU::MinorCPUPort & getDcachePort()
Returns the DcachePort owned by this Execute to pass upwards.
std::vector< bool > cantForwardFromFUIndices
FUs which this pipeline can't receive a forwarded (i.e.
friend std::ostream & operator<<(std::ostream &os, DrainState state)
bool accessesInFlight() const
Are there any accesses other than normal cached loads in the memory system or having received respons...
unsigned int issueLimit
Number of instructions that can be issued per cycle.
bool setTraceTimeOnIssue
Modify instruction trace times on issue.
Forward flowing data between Fetch2,Decode,Execute carrying a packet of instructions of a width appro...
const std::string to_string(sc_enc enc)
Queue< QueuedInst, ReportTraitsAdaptor< QueuedInst > > * inFUMemInsts
Memory ref instructions still in the FUs.
A functional unit configured from a MinorFU object.
bool isStreamChange() const
As static isStreamChange but on this branch data.
const ForwardInstData * getInput(ThreadID tid)
Get a piece of data to work on from the inputBuffer, or 0 if there is no data.
void updateBranchData(ThreadID tid, BranchData::Reason reason, MinorDynInstPtr inst, const TheISA::PCState &target, BranchData &branch)
Actually create a branch to communicate to Fetch1/Fetch2 and, if that is a stream-changing branch upd...
bool needsToTick()
May need to be ticked next cycle as one of the queues contains an actionable transfers or address tra...