42 #ifndef __CPU_CHECKER_CPU_IMPL_HH__
43 #define __CPU_CHECKER_CPU_IMPL_HH__
49 #include "config/the_isa.hh"
57 #include "debug/Checker.hh"
68 fault->invoke(tc, curStaticInst);
69 thread->decoder.reset();
72 if (curStaticInst->isLastMicroop())
76 thread->pcState(pcState);
87 DPRINTF(
Checker,
"IRQ detected at PC: %s with %d insts in buffer\n",
88 thread->
pcState(), instList.size());
90 if (!instList.empty()) {
95 for (itr = instList.begin(); itr != instList.end(); itr++) {
96 (*itr)->setCompleted();
99 inst = instList.front();
100 boundaryInst = instList.back();
104 if ((!boundaryInst && curMacroStaticInst &&
105 curStaticInst->isDelayedCommit() &&
106 !curStaticInst->isLastMicroop()) ||
107 (boundaryInst && boundaryInst->isDelayedCommit() &&
108 !boundaryInst->isLastMicroop())) {
109 panic(
"%lli: Trying to take an interrupt in middle of "
110 "a non-interuptable instruction!",
curTick());
113 thread->decoder.reset();
117 template <
class Impl>
126 if ((completed_inst->isSerializing() ||
127 completed_inst->isSerializeBefore()) &&
129 (instList.front()->seqNum != completed_inst->seqNum) : 0)) {
130 panic(
"%lli: Instruction sn:%lli at PC %s is serializing before but is"
131 " entering instList with other instructions\n",
curTick(),
132 completed_inst->seqNum, completed_inst->pcState());
140 if (!instList.empty()) {
141 if (youngestSN < completed_inst->seqNum) {
143 completed_inst->seqNum, completed_inst->
pcState());
144 instList.push_back(completed_inst);
145 youngestSN = completed_inst->seqNum;
148 if (!instList.front()->isCompleted()) {
151 inst = instList.front();
152 instList.pop_front();
155 if (!completed_inst->isCompleted()) {
156 if (youngestSN < completed_inst->seqNum) {
158 completed_inst->seqNum, completed_inst->
pcState());
159 instList.push_back(completed_inst);
160 youngestSN = completed_inst->seqNum;
164 if (youngestSN < completed_inst->seqNum) {
165 inst = completed_inst;
166 youngestSN = completed_inst->seqNum;
175 if (inst->isSerializeAfter() && !instList.empty()) {
176 panic(
"%lli: Instruction sn:%lli at PC %s is serializing after but is"
177 " exiting instList with other instructions\n",
curTick(),
178 completed_inst->seqNum, completed_inst->pcState());
180 unverifiedInst = inst;
188 unverifiedInst->seqNum, unverifiedInst->
pcState());
189 unverifiedReq = NULL;
190 unverifiedReq = unverifiedInst->reqToVerify;
191 unverifiedMemData = unverifiedInst->memData;
193 while (!result.empty()) {
196 baseStats.numCycles++;
210 if (newPCState == thread->pcState()) {
213 warn(
"%lli: Changed PC does not match expected PC, "
214 "changed: %s, expected: %s",
215 curTick(), thread->pcState(), newPCState);
218 willChangePC =
false;
224 uint64_t fetchOffset = 0;
225 bool fetchDone =
false;
228 Addr fetch_PC = thread->instAddr();
229 fetch_PC = (fetch_PC & PCMask) + fetchOffset;
234 if (!curMacroStaticInst) {
236 auto mem_req = std::make_shared<Request>(
238 fetch_PC, thread->contextId());
244 fault = mmu->translateFunctional(
248 if (unverifiedInst->getFault() ==
NoFault) {
254 warn(
"%lli: Instruction PC %s was not found in the "
255 "ITB!",
curTick(), thread->pcState());
256 handleError(unverifiedInst);
262 unverifiedInst = NULL;
268 fault = unverifiedInst->getFault();
275 icachePort->sendFunctional(pkt);
286 curStaticInst = thread->decoder.fetchRomMicroop(
287 pcState.microPC(),
nullptr);
288 }
else if (!curMacroStaticInst) {
294 Addr fetchPC = (pcState.instAddr() & PCMask) + fetchOffset;
295 thread->decoder.moreBytes(pcState, fetchPC, machInst);
300 if (thread->decoder.instReady()) {
302 instPtr = thread->decoder.decode(pcState);
303 thread->pcState(pcState);
312 curMacroStaticInst = instPtr;
316 curStaticInst = instPtr;
327 thread->decoder.reset();
331 Fault unverifiedFault;
333 unverifiedFault = unverifiedInst->getFault();
337 validateInst(unverifiedInst);
349 if (!unverifiedInst->isUnverifiable()) {
355 fault = curStaticInst->execute(
this, traceData);
363 thread->funcExeInst++;
365 validateExecution(unverifiedInst);
367 if (curStaticInst->isLoad()) {
371 panic(
"%lli: sn: %lli at PC: %s took a fault in checker "
372 "but not in driver CPU\n",
curTick(),
373 unverifiedInst->seqNum, unverifiedInst->pcState());
375 panic(
"%lli: sn: %lli at PC: %s took a fault in driver "
376 "CPU but not in checker\n",
curTick(),
377 unverifiedInst->seqNum, unverifiedInst->pcState());
384 fault->invoke(tc, curStaticInst);
386 newPCState = thread->pcState();
401 oldpc = thread->instAddr();
402 thread->pcEventQueue.service(oldpc, tc);
404 }
while (oldpc != thread->instAddr());
407 newPCState = thread->pcState();
418 if (instList.empty()) {
420 }
else if (instList.front()->isCompleted()) {
421 unverifiedInst = NULL;
422 unverifiedInst = instList.front();
423 instList.pop_front();
428 unverifiedInst = NULL;
431 template <
class Impl>
438 template <
class Impl>
444 template <
class Impl>
448 if (inst->instAddr() != thread->instAddr()) {
449 warn(
"%lli: PCs do not match! Inst: %s, checker: %s",
450 curTick(), inst->pcState(), thread->pcState());
452 warn(
"%lli: Changed PCs recently, may not be an error",
459 if (curStaticInst != inst->staticInst) {
460 warn(
"%lli: StaticInstPtrs don't match. (%s, %s).\n",
curTick(),
461 curStaticInst->getName(), inst->staticInst->getName());
465 template <
class Impl>
472 bool result_mismatch =
false;
473 bool scalar_mismatch =
false;
474 bool vector_mismatch =
false;
476 if (inst->isUnverifiable()) {
481 }
else if (inst->numDestRegs() > 0 && !result.empty()) {
482 DPRINTF(
Checker,
"Dest regs %d, number of checker dest regs %d\n",
483 inst->numDestRegs(), result.size());
484 for (
int i = 0;
i < inst->numDestRegs() && !result.empty();
i++) {
485 checker_val = result.front();
487 inst_val = inst->popResult(
489 if (checker_val != inst_val) {
490 result_mismatch =
true;
492 scalar_mismatch = checker_val.
isScalar();
493 vector_mismatch = checker_val.
isVector();
494 panic_if(!(scalar_mismatch || vector_mismatch),
495 "Unknown type of result\n");
504 if (result_mismatch) {
505 if (scalar_mismatch) {
506 warn(
"%lli: Instruction results (%i) do not match! (Values may"
507 " not actually be integers) Inst: %#x, checker: %#x",
519 if (inst->isLoad() && warnOnlyOnLoadError) {
520 copyResult(inst, inst_val, idx);
526 if (inst->nextInstAddr() != thread->nextInstAddr()) {
527 warn(
"%lli: Instruction next PCs do not match! Inst: %#x, "
529 curTick(), inst->nextInstAddr(), thread->nextInstAddr());
538 while (!miscRegIdxs.empty()) {
539 int misc_reg_idx = miscRegIdxs.front();
542 if (inst->tcBase()->readMiscRegNoEffect(misc_reg_idx) !=
543 thread->readMiscRegNoEffect(misc_reg_idx)) {
544 warn(
"%lli: Misc reg idx %i (side effect) does not match! "
545 "Inst: %#x, checker: %#x",
547 inst->tcBase()->readMiscRegNoEffect(misc_reg_idx),
548 thread->readMiscRegNoEffect(misc_reg_idx));
558 template <
class Impl>
562 if (updateThisCycle) {
564 panic(
"%lli: Instruction PC %#x results didn't match up, copying all "
565 "registers from main CPU",
curTick(), unverifiedInst->instAddr());
568 bool no_squash_from_TC = unverifiedInst->thread->noSquashFromTC;
569 unverifiedInst->thread->noSquashFromTC =
true;
572 thread->copyArchRegs(unverifiedInst->tcBase());
573 unverifiedInst->thread->noSquashFromTC = no_squash_from_TC;
576 curStaticInst = unverifiedInst->staticInst;
579 updateThisCycle =
false;
583 template <
class Impl>
586 const InstResult& mismatch_val,
int start_idx)
590 if (start_idx >= 0) {
591 const RegId& idx = inst->destRegIdx(start_idx);
603 thread->setVecReg(idx, mismatch_val.
asVector());
607 "Unexpected type of result");
624 for (
int i = start_idx;
i < inst->numDestRegs();
i++) {
625 const RegId& idx = inst->destRegIdx(
i);
626 res = inst->popResult();
638 thread->setVecReg(idx, res.
asVector());
651 thread->setMiscReg(idx.
index(), 0);
660 template <
class Impl>
664 cprintf(
"Error detected, instruction information:\n");
665 cprintf(
"PC:%s, nextPC:%#x\n[sn:%lli]\n[tid:%i]\n"
668 inst->nextInstAddr(),
671 inst->isCompleted());
676 template <
class Impl>
684 cprintf(
"Inst list size: %i\n", instList.size());
686 while (inst_list_it != instList.end())
691 cprintf(
"PC:%s\n[sn:%lli]\n[tid:%i]\n"
693 (*inst_list_it)->pcState(),
694 (*inst_list_it)->seqNum,
695 (*inst_list_it)->threadNumber,
696 (*inst_list_it)->isCompleted());
706 #endif//__CPU_CHECKER_CPU_IMPL_HH__