Hand merge
[gem5.git] / arch / alpha / isa_desc
index 5154d78d197e8cf69262809cf4dbff9275876d25..904af3ef0defac65e5aabf7a6699b9173393453a 100644 (file)
@@ -187,15 +187,16 @@ output header {{
 
        /// Print a register name for disassembly given the unique
        /// dependence tag number (FP or int).
-       void printReg(std::ostream &os, int reg);
+       void printReg(std::ostream &os, int reg) const;
 
-       std::string generateDisassembly(Addr pc, const SymbolTable *symtab);
+       std::string
+       generateDisassembly(Addr pc, const SymbolTable *symtab) const;
     };
 }};
 
 output decoder {{
     void
-    AlphaStaticInst::printReg(std::ostream &os, int reg)
+    AlphaStaticInst::printReg(std::ostream &os, int reg) const
     {
        if (reg < FP_Base_DepTag) {
            ccprintf(os, "r%d", reg);
@@ -206,7 +207,8 @@ output decoder {{
     }
 
     std::string
-    AlphaStaticInst::generateDisassembly(Addr pc, const SymbolTable *symtab)
+    AlphaStaticInst::generateDisassembly(Addr pc,
+                                        const SymbolTable *symtab) const
     {
        std::stringstream ss;
 
@@ -237,7 +239,7 @@ output decoder {{
 
 // Declarations for execute() methods.
 def template BasicExecDeclare {{
-    Fault execute(%(CPU_exec_context)s *, Trace::InstRecord *);
+    Fault execute(%(CPU_exec_context)s *, Trace::InstRecord *) const;
 }};
 
 // Basic instruction class declaration template.
@@ -267,7 +269,7 @@ def template BasicConstructor {{
 // Basic instruction class execute method template.
 def template BasicExecute {{
     Fault %(class_name)s::execute(%(CPU_exec_context)s *xc,
-                                 Trace::InstRecord *traceData)
+                                 Trace::InstRecord *traceData) const
     {
        Fault fault = No_Fault;
 
@@ -330,14 +332,16 @@ output header {{
 
        ~Nop() { }
 
-       std::string generateDisassembly(Addr pc, const SymbolTable *symtab);
+       std::string
+       generateDisassembly(Addr pc, const SymbolTable *symtab) const;
 
        %(BasicExecDeclare)s
     };
 }};
 
 output decoder {{
-    std::string Nop::generateDisassembly(Addr pc, const SymbolTable *symtab)
+    std::string Nop::generateDisassembly(Addr pc,
+                                        const SymbolTable *symtab) const
     {
 #ifdef SS_COMPATIBLE_DISASSEMBLY
        return originalDisassembly;
@@ -360,7 +364,7 @@ output decoder {{
 
 output exec {{
     Fault
-    Nop::execute(%(CPU_exec_context)s *, Trace::InstRecord *)
+    Nop::execute(%(CPU_exec_context)s *, Trace::InstRecord *) const
     {
        return No_Fault;
     }
@@ -410,13 +414,14 @@ output header {{
        {
        }
 
-       std::string generateDisassembly(Addr pc, const SymbolTable *symtab);
+       std::string
+       generateDisassembly(Addr pc, const SymbolTable *symtab) const;
     };
 }};
 
 output decoder {{
     std::string
-    IntegerImm::generateDisassembly(Addr pc, const SymbolTable *symtab)
+    IntegerImm::generateDisassembly(Addr pc, const SymbolTable *symtab) const
     {
        std::stringstream ss;
 
@@ -588,12 +593,13 @@ output header {{
        }
 
 #if defined(linux)
-       int getC99RoundingMode(uint64_t fpcr_val);
+       int getC99RoundingMode(uint64_t fpcr_val) const;
 #endif
 
        // This differs from the AlphaStaticInst version only in
        // printing suffixes for non-default rounding & trapping modes.
-       std::string generateDisassembly(Addr pc, const SymbolTable *symtab);
+       std::string
+       generateDisassembly(Addr pc, const SymbolTable *symtab) const;
     };
 
 }};
@@ -618,7 +624,7 @@ def template FloatingPointDecode {{
 output decoder {{
 #if defined(linux)
     int
-    AlphaFP::getC99RoundingMode(uint64_t fpcr_val)
+    AlphaFP::getC99RoundingMode(uint64_t fpcr_val) const
     {
        if (roundingMode == Dynamic) {
            return alphaToC99RoundingMode[bits(fpcr_val, 59, 58)];
@@ -630,7 +636,7 @@ output decoder {{
 #endif
 
     std::string
-    AlphaFP::generateDisassembly(Addr pc, const SymbolTable *symtab)
+    AlphaFP::generateDisassembly(Addr pc, const SymbolTable *symtab) const
     {
        std::string mnem_str(mnemonic);
 
@@ -738,9 +744,9 @@ output header {{
        /// Memory request flags.  See mem_req_base.hh.
         unsigned memAccessFlags;
        /// Pointer to EAComp object.
-       const StaticInstPtr<AlphaISA> eaCompPtr;
+       StaticInstPtr<AlphaISA> eaCompPtr;
        /// Pointer to MemAcc object.
-       const StaticInstPtr<AlphaISA> memAccPtr;
+       StaticInstPtr<AlphaISA> memAccPtr;
 
        /// Constructor
        Memory(const char *mnem, MachInst _machInst, OpClass __opClass,
@@ -751,12 +757,13 @@ output header {{
        {
        }
 
-       std::string generateDisassembly(Addr pc, const SymbolTable *symtab);
+       std::string
+       generateDisassembly(Addr pc, const SymbolTable *symtab) const;
 
       public:
 
-       const StaticInstPtr<AlphaISA> &eaCompInst() const { return eaCompPtr; }
-       const StaticInstPtr<AlphaISA> &memAccInst() const { return memAccPtr; }
+       StaticInstPtr<AlphaISA> &eaCompInst() { return eaCompPtr; }
+       StaticInstPtr<AlphaISA> &memAccInst() { return memAccPtr; }
     };
 
     /**
@@ -790,81 +797,33 @@ output header {{
       protected:
        /// Constructor
        MemoryNoDisp(const char *mnem, MachInst _machInst, OpClass __opClass,
-                    StaticInstPtr<AlphaISA> _eaCompPtr,
-                    StaticInstPtr<AlphaISA> _memAccPtr)
+                    StaticInstPtr<AlphaISA> _eaCompPtr = nullStaticInstPtr,
+                    StaticInstPtr<AlphaISA> _memAccPtr = nullStaticInstPtr)
            : Memory(mnem, _machInst, __opClass, _eaCompPtr, _memAccPtr)
        {
        }
 
-       std::string generateDisassembly(Addr pc, const SymbolTable *symtab);
-    };
-
-
-    /**
-     * Base class for "fake" effective-address computation
-     * instructions returnded by eaCompInst().
-     */
-    class EACompBase : public AlphaStaticInst
-    {
-      public:
-       /// Constructor
-       EACompBase(MachInst machInst)
-           : AlphaStaticInst("(eacomp)", machInst, IntAluOp)
-       {
-       }
-
-       %(BasicExecDeclare)s
-    };
-
-    /**
-     * Base class for "fake" memory-access instructions returnded by
-     * memAccInst().
-     */
-    class MemAccBase : public AlphaStaticInst
-    {
-      public:
-       /// Constructor
-       MemAccBase(MachInst machInst, OpClass __opClass)
-           : AlphaStaticInst("(memacc)", machInst, __opClass)
-       {
-       }
-
-       %(BasicExecDeclare)s
+       std::string
+       generateDisassembly(Addr pc, const SymbolTable *symtab) const;
     };
-
 }};
 
 
 output decoder {{
     std::string
-    Memory::generateDisassembly(Addr pc, const SymbolTable *symtab)
+    Memory::generateDisassembly(Addr pc, const SymbolTable *symtab) const
     {
        return csprintf("%-10s %c%d,%d(r%d)", mnemonic,
                        flags[IsFloating] ? 'f' : 'r', RA, MEMDISP, RB);
     }
 
     std::string
-    MemoryNoDisp::generateDisassembly(Addr pc, const SymbolTable *symtab)
+    MemoryNoDisp::generateDisassembly(Addr pc, const SymbolTable *symtab) const
     {
        return csprintf("%-10s (r%d)", mnemonic, RB);
     }
 }};
 
-output exec {{
-    Fault
-    EACompBase::execute(%(CPU_exec_context)s *, Trace::InstRecord *)
-    {
-       panic("attempt to execute eacomp");
-    }
-
-    Fault
-    MemAccBase::execute(%(CPU_exec_context)s *, Trace::InstRecord *)
-    {
-       panic("attempt to execute memacc");
-    }
-}};
-
-
 def format LoadAddress(code) {{
     iop = InstObjParams(name, Name, 'MemoryDisp32', CodeBlock(code))
     header_output = BasicDeclare.subst(iop)
@@ -885,21 +844,25 @@ def template LoadStoreDeclare {{
        /**
         * "Fake" effective address computation class for "%(mnemonic)s".
         */
-       class EAComp : public EACompBase
+       class EAComp : public %(base_class)s
        {
          public:
            /// Constructor
            EAComp(MachInst machInst);
+
+            %(BasicExecDeclare)s
        };
 
        /**
         * "Fake" memory access instruction class for "%(mnemonic)s".
         */
-       class MemAcc : public MemAccBase
+       class MemAcc : public %(base_class)s
        {
          public:
            /// Constructor
            MemAcc(MachInst machInst);
+
+            %(BasicExecDeclare)s
        };
 
       public:
@@ -912,14 +875,17 @@ def template LoadStoreDeclare {{
 }};
 
 def template LoadStoreConstructor {{
+    /** TODO: change op_class to AddrGenOp or something (requires
+     * creating new member of OpClass enum in op_class.hh, updating
+     * config files, etc.). */
     inline %(class_name)s::EAComp::EAComp(MachInst machInst)
-       : EACompBase(machInst)
+       : %(base_class)s("%(mnemonic)s (EAComp)", machInst, IntAluOp)
     {
        %(ea_constructor)s;
     }
 
     inline %(class_name)s::MemAcc::MemAcc(MachInst machInst)
-       : MemAccBase(machInst, %(op_class)s)
+       : %(base_class)s("%(mnemonic)s (MemAcc)", machInst, %(op_class)s)
     {
        %(memacc_constructor)s;
     }
@@ -932,9 +898,67 @@ def template LoadStoreConstructor {{
     }
 }};
 
+
+def template EACompExecute {{
+    Fault
+    %(class_name)s::EAComp::execute(%(CPU_exec_context)s *xc,
+                                  Trace::InstRecord *traceData) const
+    {
+       Addr EA;
+       Fault fault = No_Fault;
+
+       %(fp_enable_check)s;
+       %(op_decl)s;
+       %(op_rd)s;
+       %(code)s;
+
+       if (fault == No_Fault) {
+           %(op_wb)s;
+           xc->setEA(EA);
+       }
+
+       return fault;
+    }
+}};
+
+def template MemAccExecute {{
+    Fault
+    %(class_name)s::MemAcc::execute(%(CPU_exec_context)s *xc,
+                                  Trace::InstRecord *traceData) const
+    {
+       Addr EA;
+       Fault fault = No_Fault;
+
+       %(fp_enable_check)s;
+       %(op_decl)s;
+       %(op_nonmem_rd)s;
+       EA = xc->getEA();
+       
+       if (fault == No_Fault) {
+           %(op_mem_rd)s;
+           %(code)s;
+       }
+
+       if (fault == No_Fault) {
+           %(op_mem_wb)s;
+       }
+
+       if (fault == No_Fault) {
+           %(postacc_code)s;
+       }
+
+       if (fault == No_Fault) {
+           %(op_nonmem_wb)s;
+       }
+
+       return fault;
+    }
+}};
+
+
 def template LoadStoreExecute {{
     Fault %(class_name)s::execute(%(CPU_exec_context)s *xc,
-                                 Trace::InstRecord *traceData)
+                                 Trace::InstRecord *traceData) const
     {
        Addr EA;
        Fault fault = No_Fault;
@@ -968,7 +992,7 @@ def template LoadStoreExecute {{
 
 def template PrefetchExecute {{
     Fault %(class_name)s::execute(%(CPU_exec_context)s *xc,
-                                 Trace::InstRecord *traceData)
+                                 Trace::InstRecord *traceData) const
     {
        Addr EA;
        Fault fault = No_Fault;
@@ -1022,18 +1046,33 @@ def LoadStoreBase(name, Name, ea_code, memacc_code, postacc_code = '',
 
     # Would be nice to autogenerate this list, but oh well.
     valid_mem_flags = ['LOCKED', 'NO_FAULT', 'EVICT_NEXT', 'PF_EXCLUSIVE']
-    inst_flags = []
-    mem_flags = []
-    for f in flags:
-       if f in valid_mem_flags:
-           mem_flags.append(f)
-        else:
-            inst_flags.append(f)
+    mem_flags =  [f for f in flags if f in valid_mem_flags]
+    inst_flags = [f for f in flags if f not in valid_mem_flags]
+
+    # add hook to get effective addresses into execution trace output.
+    ea_code += '\nif (traceData) { traceData->setAddr(EA); }\n'
 
+    # generate code block objects
     ea_cblk = CodeBlock(ea_code)
     memacc_cblk = CodeBlock(memacc_code)
     postacc_cblk = CodeBlock(postacc_code)
 
+    # Some CPU models execute the memory operation as an atomic unit,
+    # while others want to separate them into an effective address
+    # computation and a memory access operation.  As a result, we need
+    # to generate three StaticInst objects.  Note that the latter two
+    # are nested inside the larger "atomic" one.
+
+    # generate InstObjParams for EAComp object
+    ea_iop = InstObjParams(name, Name, base_class, ea_cblk, inst_flags)
+    
+    # generate InstObjParams for MemAcc object
+    memacc_iop = InstObjParams(name, Name, base_class, memacc_cblk, inst_flags)
+    # in the split execution model, the MemAcc portion is responsible
+    # for the post-access code.
+    memacc_iop.postacc_code = postacc_cblk.code
+
+    # generate InstObjParams for unified execution
     cblk = CodeBlock(ea_code + memacc_code + postacc_code)
     iop = InstObjParams(name, Name, base_class, cblk, inst_flags)
 
@@ -1043,13 +1082,17 @@ def LoadStoreBase(name, Name, ea_code, memacc_code, postacc_code = '',
     iop.memacc_code = memacc_cblk.code
     iop.postacc_code = postacc_cblk.code
 
-    mem_flags = string.join(mem_flags, '|')
-    if mem_flags != '':
-        iop.constructor += '\n\tmemAccessFlags = ' + mem_flags + ';'
+    if mem_flags:
+        s = '\n\tmemAccessFlags = ' + string.join(mem_flags, '|') + ';'
+        iop.constructor += s
+        memacc_iop.constructor += s
 
     # (header_output, decoder_output, decode_block, exec_output)
     return (LoadStoreDeclare.subst(iop), LoadStoreConstructor.subst(iop),
-           decode_template.subst(iop), exec_template.subst(iop))
+           decode_template.subst(iop),
+            EACompExecute.subst(ea_iop)
+            + MemAccExecute.subst(memacc_iop)
+            + exec_template.subst(iop))
 }};
 
 
@@ -1123,9 +1166,9 @@ output header {{
     {
       protected:
        /// Cached program counter from last disassembly
-       Addr               cachedPC;
+       mutable Addr cachedPC;
        /// Cached symbol table pointer from last disassembly
-       const SymbolTable *cachedSymtab;
+       mutable const SymbolTable *cachedSymtab;
 
        /// Constructor
        PCDependentDisassembly(const char *mnem, MachInst _machInst,
@@ -1135,7 +1178,8 @@ output header {{
        {
        }
 
-       const std::string &disassemble(Addr pc, const SymbolTable *symtab);
+       const std::string &
+       disassemble(Addr pc, const SymbolTable *symtab) const;
     };
 
     /**
@@ -1157,7 +1201,8 @@ output header {{
 
        Addr branchTarget(Addr branchPC) const;
 
-       std::string generateDisassembly(Addr pc, const SymbolTable *symtab);
+       std::string
+       generateDisassembly(Addr pc, const SymbolTable *symtab) const;
     };
 
     /**
@@ -1181,7 +1226,8 @@ output header {{
 
        Addr branchTarget(ExecContext *xc) const;
 
-       std::string generateDisassembly(Addr pc, const SymbolTable *symtab);
+       std::string
+       generateDisassembly(Addr pc, const SymbolTable *symtab) const;
     };
 }};
 
@@ -1201,7 +1247,8 @@ output decoder {{
     }
 
     const std::string &
-    PCDependentDisassembly::disassemble(Addr pc, const SymbolTable *symtab)
+    PCDependentDisassembly::disassemble(Addr pc,
+                                       const SymbolTable *symtab) const
     {
        if (!cachedDisassembly ||
            pc != cachedPC || symtab != cachedSymtab)
@@ -1219,7 +1266,7 @@ output decoder {{
     }
 
     std::string
-    Branch::generateDisassembly(Addr pc, const SymbolTable *symtab)
+    Branch::generateDisassembly(Addr pc, const SymbolTable *symtab) const
     {
        std::stringstream ss;
 
@@ -1257,7 +1304,7 @@ output decoder {{
     }
 
     std::string
-    Jump::generateDisassembly(Addr pc, const SymbolTable *symtab)
+    Jump::generateDisassembly(Addr pc, const SymbolTable *symtab) const
     {
        std::stringstream ss;
 
@@ -1356,13 +1403,15 @@ output header {{
        {
        }
 
-       std::string generateDisassembly(Addr pc, const SymbolTable *symtab);
+       std::string
+       generateDisassembly(Addr pc, const SymbolTable *symtab) const;
     };
 }};
 
 output decoder {{
     std::string
-    EmulatedCallPal::generateDisassembly(Addr pc, const SymbolTable *symtab)
+    EmulatedCallPal::generateDisassembly(Addr pc,
+                                        const SymbolTable *symtab) const
     {
 #ifdef SS_COMPATIBLE_DISASSEMBLY
        return csprintf("%s %s", "call_pal", mnemonic);
@@ -1398,7 +1447,8 @@ output header {{
        CallPalBase(const char *mnem, MachInst _machInst,
                    OpClass __opClass);
 
-       std::string generateDisassembly(Addr pc, const SymbolTable *symtab);
+       std::string
+       generateDisassembly(Addr pc, const SymbolTable *symtab) const;
     };
 }};
 
@@ -1428,7 +1478,7 @@ output decoder {{
     }
 
     std::string
-    CallPalBase::generateDisassembly(Addr pc, const SymbolTable *symtab)
+    CallPalBase::generateDisassembly(Addr pc, const SymbolTable *symtab) const
     {
        return csprintf("%-10s %#x", "call_pal", palFunc);
     }
@@ -1460,10 +1510,11 @@ output header {{
 
        /// Constructor
        HwLoadStore(const char *mnem, MachInst _machInst, OpClass __opClass,
-                   StaticInstPtr<AlphaISA> _eaCompPtr,
-                   StaticInstPtr<AlphaISA> _memAccPtr);
+                   StaticInstPtr<AlphaISA> _eaCompPtr = nullStaticInstPtr,
+                   StaticInstPtr<AlphaISA> _memAccPtr = nullStaticInstPtr);
 
-       std::string generateDisassembly(Addr pc, const SymbolTable *symtab);
+       std::string
+       generateDisassembly(Addr pc, const SymbolTable *symtab) const;
     };
 }};
 
@@ -1485,7 +1536,7 @@ output decoder {{
     }
 
     std::string
-    HwLoadStore::generateDisassembly(Addr pc, const SymbolTable *symtab)
+    HwLoadStore::generateDisassembly(Addr pc, const SymbolTable *symtab) const
     {
 #ifdef SS_COMPATIBLE_DISASSEMBLY
        return csprintf("%-10s r%d,%d(r%d)", mnemonic, RA, disp, RB);
@@ -1536,13 +1587,14 @@ output header {{
        {
        }
 
-       std::string generateDisassembly(Addr pc, const SymbolTable *symtab);
+       std::string
+       generateDisassembly(Addr pc, const SymbolTable *symtab) const;
     };
 }};
 
 output decoder {{
     std::string
-    HwMoveIPR::generateDisassembly(Addr pc, const SymbolTable *symtab)
+    HwMoveIPR::generateDisassembly(Addr pc, const SymbolTable *symtab) const
     {
        if (_numSrcRegs > 0) {
            // must be mtpr
@@ -1558,7 +1610,8 @@ output decoder {{
 }};
 
 def format HwMoveIPR(code) {{
-    iop = InstObjParams(name, Name, 'HwMoveIPR', CodeBlock(code))
+    iop = InstObjParams(name, Name, 'HwMoveIPR', CodeBlock(code),
+                       ['IprAccessOp'])
     header_output = BasicDeclare.subst(iop)
     decoder_output = BasicConstructor.subst(iop)
     decode_block = BasicDecode.subst(iop)
@@ -1593,7 +1646,8 @@ output header {{
 
        %(BasicExecDeclare)s
 
-       std::string generateDisassembly(Addr pc, const SymbolTable *symtab);
+       std::string
+       generateDisassembly(Addr pc, const SymbolTable *symtab) const;
     };
 
     /**
@@ -1609,7 +1663,7 @@ output header {{
     {
       private:
        /// Have we warned on this instruction yet?
-       bool warned;
+       mutable bool warned;
 
       public:
        /// Constructor
@@ -1623,19 +1677,22 @@ output header {{
 
        %(BasicExecDeclare)s
 
-       std::string generateDisassembly(Addr pc, const SymbolTable *symtab);
+       std::string
+       generateDisassembly(Addr pc, const SymbolTable *symtab) const;
     };
 }};
 
 output decoder {{
     std::string
-    FailUnimplemented::generateDisassembly(Addr pc, const SymbolTable *symtab)
+    FailUnimplemented::generateDisassembly(Addr pc,
+                                          const SymbolTable *symtab) const
     {
        return csprintf("%-10s (unimplemented)", mnemonic);
     }
 
     std::string
-    WarnUnimplemented::generateDisassembly(Addr pc, const SymbolTable *symtab)
+    WarnUnimplemented::generateDisassembly(Addr pc,
+                                          const SymbolTable *symtab) const
     {
 #ifdef SS_COMPATIBLE_DISASSEMBLY
        return csprintf("%-10s", mnemonic);
@@ -1648,7 +1705,7 @@ output decoder {{
 output exec {{
     Fault
     FailUnimplemented::execute(%(CPU_exec_context)s *xc,
-                              Trace::InstRecord *traceData)
+                              Trace::InstRecord *traceData) const
     {
        panic("attempt to execute unimplemented instruction '%s' "
              "(inst 0x%08x, opcode 0x%x)", mnemonic, machInst, OPCODE);
@@ -1657,7 +1714,7 @@ output exec {{
 
     Fault
     WarnUnimplemented::execute(%(CPU_exec_context)s *xc,
-                              Trace::InstRecord *traceData)
+                              Trace::InstRecord *traceData) const
     {
        if (!warned) {
            warn("instruction '%s' unimplemented\n", mnemonic);
@@ -1699,7 +1756,8 @@ output header {{
 
        %(BasicExecDeclare)s
 
-       std::string generateDisassembly(Addr pc, const SymbolTable *symtab);
+       std::string
+       generateDisassembly(Addr pc, const SymbolTable *symtab) const;
     };
 }};
 
@@ -1710,7 +1768,7 @@ output header {{
 
 output decoder {{
     std::string
-    Unknown::generateDisassembly(Addr pc, const SymbolTable *symtab)
+    Unknown::generateDisassembly(Addr pc, const SymbolTable *symtab) const
     {
        return csprintf("%-10s (inst 0x%x, opcode 0x%x)",
                        "unknown", machInst, OPCODE);
@@ -1719,7 +1777,8 @@ output decoder {{
 
 output exec {{
     Fault
-    Unknown::execute(%(CPU_exec_context)s *xc, Trace::InstRecord *traceData)
+    Unknown::execute(%(CPU_exec_context)s *xc,
+                    Trace::InstRecord *traceData) const
     {
        panic("attempt to execute unknown instruction "
              "(inst 0x%08x, opcode 0x%x)", machInst, OPCODE);
@@ -2480,9 +2539,9 @@ decode OPCODE default Unknown::unknown() {
                xc->syscall();
            }}, IsNonSpeculative);
            // Read uniq reg into ABI return value register (r0)
-           0x9e: rduniq({{ R0 = Runiq; }});
+           0x9e: rduniq({{ R0 = Runiq; }}, IsNonSpeculative);
            // Write uniq reg with value from ABI arg register (r16)
-           0x9f: wruniq({{ Runiq = R16; }});
+           0x9f: wruniq({{ Runiq = R16; }}, IsNonSpeculative);
        }
     }
 #endif
@@ -2507,7 +2566,7 @@ decode OPCODE default Unknown::unknown() {
     }
 
     format BasicOperate {
-       0x1e: hw_rei({{ xc->hwrei(); }});
+       0x1e: hw_rei({{ xc->hwrei(); }}, IsSerializing);
 
        // M5 special opcodes use the reserved 0x01 opcode space
        0x01: decode M5FUNC {