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__