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__