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"
65 template <
class DynInstPtr>
71 fault->invoke(tc, curStaticInst);
72 thread->decoder->reset();
75 if (curStaticInst->isLastMicroop())
84 template <
class DynInstPtr>
88 DPRINTF(
Checker,
"IRQ detected at PC: %s with %d insts in buffer\n",
89 thread->
pcState(), instList.size());
91 if (!instList.empty()) {
96 for (itr = instList.begin(); itr != instList.end(); itr++) {
97 (*itr)->setCompleted();
100 inst = instList.front();
101 boundaryInst = instList.back();
105 if ((!boundaryInst && curMacroStaticInst &&
106 curStaticInst->isDelayedCommit() &&
107 !curStaticInst->isLastMicroop()) ||
108 (boundaryInst && boundaryInst->isDelayedCommit() &&
109 !boundaryInst->isLastMicroop())) {
110 panic(
"%lli: Trying to take an interrupt in middle of "
111 "a non-interuptable instruction!",
curTick());
114 thread->decoder->reset();
118 template <
class DynInstPtr>
127 if ((completed_inst->isSerializing() ||
128 completed_inst->isSerializeBefore()) &&
130 (instList.front()->seqNum != completed_inst->seqNum) : 0)) {
131 panic(
"%lli: Instruction sn:%lli at PC %s is serializing before but is"
132 " entering instList with other instructions\n",
curTick(),
133 completed_inst->seqNum, completed_inst->pcState());
141 if (!instList.empty()) {
142 if (youngestSN < completed_inst->seqNum) {
144 completed_inst->seqNum, completed_inst->
pcState());
145 instList.push_back(completed_inst);
146 youngestSN = completed_inst->seqNum;
149 if (!instList.front()->isCompleted()) {
152 inst = instList.front();
153 instList.pop_front();
156 if (!completed_inst->isCompleted()) {
157 if (youngestSN < completed_inst->seqNum) {
159 completed_inst->seqNum, completed_inst->
pcState());
160 instList.push_back(completed_inst);
161 youngestSN = completed_inst->seqNum;
165 if (youngestSN < completed_inst->seqNum) {
166 inst = completed_inst;
167 youngestSN = completed_inst->seqNum;
176 if (inst->isSerializeAfter() && !instList.empty()) {
177 panic(
"%lli: Instruction sn:%lli at PC %s is serializing after but is"
178 " exiting instList with other instructions\n",
curTick(),
179 completed_inst->seqNum, completed_inst->pcState());
181 unverifiedInst = inst;
184 auto &
decoder = thread->decoder;
192 unverifiedInst->seqNum, unverifiedInst->
pcState());
193 unverifiedReq = NULL;
194 unverifiedReq = unverifiedInst->reqToVerify;
195 unverifiedMemData = unverifiedInst->memData;
197 while (!result.empty()) {
200 baseStats.numCycles++;
211 if (*newPCState == thread->pcState()) {
214 warn(
"%lli: Changed PC does not match expected PC, "
215 "changed: %s, expected: %s",
216 curTick(), thread->pcState(), *newPCState);
219 willChangePC =
false;
225 uint64_t fetchOffset = 0;
226 bool fetchDone =
false;
228 Addr fetch_PC = thread->pcState().instAddr();
229 fetch_PC = (fetch_PC & pc_mask) + fetchOffset;
232 if (!curMacroStaticInst) {
234 auto mem_req = std::make_shared<Request>(
235 fetch_PC,
decoder->moreBytesSize(), 0, requestorId,
236 fetch_PC, thread->contextId());
238 mem_req->setVirt(fetch_PC,
decoder->moreBytesSize(),
240 thread->pcState().instAddr());
242 fault = mmu->translateFunctional(
246 if (unverifiedInst->getFault() ==
NoFault) {
252 warn(
"%lli: Instruction PC %s was not found in the "
253 "ITB!",
curTick(), thread->pcState());
254 handleError(unverifiedInst);
260 unverifiedInst = NULL;
266 fault = unverifiedInst->getFault();
273 icachePort->sendFunctional(pkt);
280 std::unique_ptr<PCStateBase> pc_state(
281 thread->pcState().clone());
285 curStaticInst =
decoder->fetchRomMicroop(
286 pc_state->microPC(),
nullptr);
287 }
else if (!curMacroStaticInst) {
294 (pc_state->instAddr() & pc_mask) + fetchOffset;
295 decoder->moreBytes(*pc_state, fetch_pc);
302 instPtr =
decoder->decode(*pc_state);
303 thread->pcState(*pc_state);
306 fetchOffset +=
decoder->moreBytesSize();
312 curMacroStaticInst = instPtr;
316 curStaticInst = instPtr;
331 Fault unverifiedFault;
333 unverifiedFault = unverifiedInst->getFault();
337 validateInst(unverifiedInst);
349 if (!unverifiedInst->isUnverifiable()) {
355 fault = curStaticInst->execute(
this, traceData);
364 validateExecution(unverifiedInst);
366 if (curStaticInst->isLoad()) {
370 panic(
"%lli: sn: %lli at PC: %s took a fault in checker "
371 "but not in driver CPU\n",
curTick(),
372 unverifiedInst->seqNum, unverifiedInst->pcState());
374 panic(
"%lli: sn: %lli at PC: %s took a fault in driver "
375 "CPU but not in checker\n",
curTick(),
376 unverifiedInst->seqNum, unverifiedInst->pcState());
383 fault->invoke(tc, curStaticInst);
385 set(newPCState, thread->pcState());
400 oldpc = thread->pcState().instAddr();
401 thread->pcEventQueue.service(oldpc, tc);
403 }
while (oldpc != thread->pcState().instAddr());
406 set(newPCState, thread->pcState());
417 if (instList.empty()) {
419 }
else if (instList.front()->isCompleted()) {
420 unverifiedInst = NULL;
421 unverifiedInst = instList.front();
422 instList.pop_front();
427 unverifiedInst = NULL;
430 template <
class DynInstPtr>
437 template <
class DynInstPtr>
440 template <
class DynInstPtr>
444 if (inst->pcState().instAddr() != thread->pcState().instAddr()) {
445 warn(
"%lli: PCs do not match! Inst: %s, checker: %s",
446 curTick(), inst->pcState(), thread->pcState());
448 warn(
"%lli: Changed PCs recently, may not be an error",
455 if (curStaticInst != inst->staticInst) {
456 warn(
"%lli: StaticInstPtrs don't match. (%s, %s).\n",
curTick(),
457 curStaticInst->getName(), inst->staticInst->getName());
461 template <
class DynInstPtr>
468 bool result_mismatch =
false;
469 bool scalar_mismatch =
false;
471 if (inst->isUnverifiable()) {
476 }
else if (inst->numDestRegs() > 0 && !result.empty()) {
477 DPRINTF(
Checker,
"Dest regs %d, number of checker dest regs %d\n",
478 inst->numDestRegs(), result.size());
479 for (
int i = 0;
i < inst->numDestRegs() && !result.empty();
i++) {
480 checker_val = result.front();
483 if (checker_val != inst_val) {
484 result_mismatch =
true;
486 scalar_mismatch = checker_val.
is<
RegVal>();
495 if (result_mismatch) {
496 if (scalar_mismatch) {
497 warn(
"%lli: Instruction results (%i) do not match! (Values may"
498 " not actually be integers) Inst: %#x, checker: %#x",
510 if (inst->isLoad() && warnOnlyOnLoadError) {
511 copyResult(inst, inst_val, idx);
517 if (inst->pcState() != thread->pcState()) {
518 warn(
"%lli: Instruction PCs do not match! Inst: %s, checker: %s",
519 curTick(), inst->pcState(), thread->pcState());
528 while (!miscRegIdxs.empty()) {
529 int misc_reg_idx = miscRegIdxs.front();
532 if (inst->tcBase()->readMiscRegNoEffect(misc_reg_idx) !=
533 thread->readMiscRegNoEffect(misc_reg_idx)) {
534 warn(
"%lli: Misc reg idx %i (side effect) does not match! "
535 "Inst: %#x, checker: %#x",
537 inst->tcBase()->readMiscRegNoEffect(misc_reg_idx),
538 thread->readMiscRegNoEffect(misc_reg_idx));
548 template <
class DynInstPtr>
552 if (updateThisCycle) {
554 panic(
"%lli: Instruction PC %#x results didn't match up, copying all "
555 "registers from main CPU",
curTick(),
556 unverifiedInst->pcState().instAddr());
559 bool no_squash_from_TC = unverifiedInst->thread->noSquashFromTC;
560 unverifiedInst->thread->noSquashFromTC =
true;
563 thread->copyArchRegs(unverifiedInst->tcBase());
564 unverifiedInst->thread->noSquashFromTC = no_squash_from_TC;
567 curStaticInst = unverifiedInst->staticInst;
570 updateThisCycle =
false;
574 template <
class DynInstPtr>
581 if (start_idx >= 0) {
582 const RegId& idx = inst->destRegIdx(start_idx);
590 thread->setReg(idx, mismatch_val.
as<
RegVal>());
595 thread->setReg(idx, &
val);
607 for (
int i = start_idx;
i < inst->numDestRegs();
i++) {
608 const RegId& idx = inst->destRegIdx(
i);
609 res = inst->popResult();
617 thread->setReg(idx, res.
as<
RegVal>());
622 thread->setReg(idx, &
val);
627 thread->setMiscReg(idx.
index(), 0);
636 template <
class DynInstPtr>
640 cprintf(
"Error detected, instruction information:\n");
641 cprintf(
"PC:%s\n[sn:%lli]\n[tid:%i]\n"
646 inst->isCompleted());
651 template <
class DynInstPtr>
659 cprintf(
"Inst list size: %i\n", instList.size());
661 while (inst_list_it != instList.end())
666 cprintf(
"PC:%s\n[sn:%lli]\n[tid:%i]\n"
668 (*inst_list_it)->pcState(),
669 (*inst_list_it)->seqNum,
670 (*inst_list_it)->threadNumber,
671 (*inst_list_it)->isCompleted());
683 #endif//__CPU_CHECKER_CPU_IMPL_HH__