我刚刚看了这段视频http://www.youtube.com/watch?v=y71lli8MS8s,发现它很棒。问题是关于他的操作码解码矩阵函数:

template<u16 op> // Execute a single CPU instruction, defined by opcode "op".
void Ins()       // With template magic, the compiler will literally synthesize >256 different functions.
{
    // Note: op 0x100 means "NMI", 0x101 means "Reset", 0x102 means "IRQ". They are implemented in terms of "BRK".
    // User is responsible for ensuring that WB() will not store into memory while Reset is being processed.
    unsigned addr=0, d=0, t=0xFF, c=0, sb=0, pbits = op<0x100 ? 0x30 : 0x20;

    // Define the opcode decoding matrix, which decides which micro-operations constitute
    // any particular opcode. (Note: The PLA of 6502 works on a slightly different principle.)
    enum { o8 = op/8, o8m = 1 << (op%8) };
    // Fetch op'th item from a bitstring encoded in a data-specific variant of base64,
    // where each character transmits 8 bits of information rather than 6.
    // This peculiar encoding was chosen to reduce the source code size.
    // Enum temporaries are used in order to ensure compile-time evaluation.
    #define t(s,code) { enum { \
        i=o8m & (s[o8]>90 ? (130+" (),-089<>?BCFGHJLSVWZ[^hlmnxy|}"[s[o8]-94]) \
                          : (s[o8]-" (("[s[o8]/39])) }; if(i) { code; } }

    /* Decode address operand */
    t("                                !", addr = 0xFFFA) // NMI vector location
    t("                                *", addr = 0xFFFC) // Reset vector location
    t("!                               ,", addr = 0xFFFE) // Interrupt vector location
    t("zy}z{y}zzy}zzy}zzy}zzy}zzy}zzy}z ", addr = RB(PC++))
    t("2 yy2 yy2 yy2 yy2 XX2 XX2 yy2 yy ", d = X) // register index
    t("  62  62  62  62  om  om  62  62 ", d = Y)
    t("2 y 2 y 2 y 2 y 2 y 2 y 2 y 2 y  ", addr=u8(addr+d); d=0; tick())              // add zeropage-index
    t(" y z!y z y z y z y z y z y z y z ", addr=u8(addr);   addr+=256*RB(PC++))       // absolute address
    t("3 6 2 6 2 6 286 2 6 2 6 2 6 2 6 /", addr=RB(c=addr); addr+=256*RB(wrap(c,c+1)))// indirect w/ page wrap
    t("  *Z  *Z  *Z  *Z      6z  *Z  *Z ", Misfire(addr, addr+d)) // abs. load: extra misread when cross-page
    t("  4k  4k  4k  4k  6z      4k  4k ", RB(wrap(addr, addr+d)))// abs. store: always issue a misread
    /* Load source operand */
    t("aa__ff__ab__,4  ____ -  ____     ", t &= A) // Many operations take A or X as operand. Some try in
    t("                knnn     4  99   ", t &= X) // error to take both; the outcome is an AND operation.
    t("                9989    99       ", t &= Y) // sty,dey,iny,tya,cpy
    t("                       4         ", t &= S) // tsx, las
    t("!!!!  !!  !!  !!  !   !!  !!  !!/", t &= P.raw|pbits; c = t)// php, flag test/set/clear, interrupts
    t("_^__dc___^__            ed__98   ", c = t; t = 0xFF)        // save as second operand
    t("vuwvzywvvuwvvuwv    zy|zzywvzywv ", t &= RB(addr+d)) // memory operand
    t(",2  ,2  ,2  ,2  -2  -2  -2  -2   ", t &= RB(PC++))   // immediate operand
    /* Operations that mogrify memory operands directly */
    t("    88                           ", P.V = t & 0x40; P.N = t & 0x80) // bit
    t("    nink    nnnk                 ", sb = P.C)       // rol,rla, ror,rra,arr
    t("nnnknnnk     0                   ", P.C = t & 0x80) // rol,rla, asl,slo,[arr,anc]
    t("        nnnknink                 ", P.C = t & 0x01) // lsr,sre, ror,rra,asr
    t("ninknink                         ", t = (t << 1) | (sb * 0x01))
    t("        nnnknnnk                 ", t = (t >> 1) | (sb * 0x80))
    t("                 !      kink     ", t = u8(t - 1))  // dec,dex,dey,dcp
    t("                         !  khnk ", t = u8(t + 1))  // inc,inx,iny,isb
    /* Store modified value (memory) */
    t("kgnkkgnkkgnkkgnkzy|J    kgnkkgnk ", WB(addr+d, t))
    t("                   q             ", WB(wrap(addr, addr+d), t &= ((addr+d) >> 8))) // [shx,shy,shs,sha?]
    /* Some operations used up one clock cycle that we did not account for yet */
    t("rpstljstqjstrjst - - - -kjstkjst/", tick()) // nop,flag ops,inc,dec,shifts,stack,transregister,interrupts
    /* Stack operations and unconditional jumps */
    t("     !  !    !                   ", tick(); t = Pop())                        // pla,plp,rti
    t("        !   !                    ", RB(PC++); PC = Pop(); PC |= (Pop() << 8)) // rti,rts
    t("            !                    ", RB(PC++))  // rts
    t("!   !                           /", d=PC+(op?-1:1); Push(d>>8); Push(d))      // jsr, interrupts
    t("!   !    8   8                  /", PC = addr) // jmp, jsr, interrupts
    t("!!       !                      /", Push(t))   // pha, php, interrupts
    /* Bitmasks */
    t("! !!  !!  !!  !!  !   !!  !!  !!/", t = 1)
    t("  !   !                   !!  !! ", t <<= 1)
    t("! !   !   !!  !!       !   !   !/", t <<= 2)
    t("  !   !   !   !        !         ", t <<= 4)
    t("   !       !           !   !____ ", t = u8(~t)) // sbc, isb,      clear flag
    t("`^__   !       !               !/", t = c | t)  // ora, slo,      set flag
    t("  !!dc`_  !!  !   !   !!  !!  !  ", t = c & t)  // and, bit, rla, clear/test flag
    t("        _^__                     ", t = c ^ t)  // eor, sre
    /* Conditional branches */
    t("      !       !       !       !  ", if(t)  { tick(); Misfire(PC, addr = s8(addr) + PC); PC=addr; })
    t("  !       !       !       !      ", if(!t) { tick(); Misfire(PC, addr = s8(addr) + PC); PC=addr; })
    /* Addition and subtraction */
    t("            _^__            ____ ", c = t; t += A + P.C; P.V = (c^t) & (A^t) & 0x80; P.C = t & 0x100)
    t("                        ed__98   ", t = c - t; P.C = ~t & 0x100) // cmp,cpx,cpy, dcp, sbx
    /* Store modified value (register) */
    t("aa__aa__aa__ab__ 4 !____    ____ ", A = t)
    t("                    nnnn 4   !   ", X = t) // ldx, dex, tax, inx, tsx,lax,las,sbx
    t("                 !  9988 !       ", Y = t) // ldy, dey, tay, iny
    t("                   4   0         ", S = t) // txs, las, shs
    t("!  ! ! !!  !   !       !   !   !/", P.raw = t & ~0x30) // plp, rti, flag set/clear
    /* Generic status flag updates */
    t("wwwvwwwvwwwvwxwv 5 !}}||{}wv{{wv ", P.N = t & 0x80)
    t("wwwv||wvwwwvwxwv 5 !}}||{}wv{{wv ", P.Z = u8(t) == 0)
    t("             0                   ", P.V = (((t >> 5)+1)&2))         // [arr]
    /* All implemented opcodes are cycle-accurate and memory-access-accurate.
     * [] means that this particular separate rule exists only to provide the indicated unofficial opcode(s).
     */
}

我感到非常困惑的是define语句。宏展开后,它看起来像http://codepad.org/bUxdX8MQ。但是'130+" (),-089<>?BCFGHJLSVWZ[^hlmnxy|}"'如何成为合法的C++代码?据我所知,以这种方式组合字符串和整数是非法的。同样,除非C++ 11添加了对以数组方式访问字符串字符的支持,否则("zy}z{y}zzy}zzy}zzy}zzy}zzy}zzy}z "[o8]>90)根本没有任何意义。希望学习新的东西!

编辑:谢谢所有答复的人。我不知道现在C / C++中允许"blabla"[idx]了,因为我这样做是有道理的。

最佳答案

实际上应该是

130+" (),-089<>?BCFGHJLSVWZ[^hlmnxy|}"["                                !"[o8]-94]

从第一个C++规范开始,字符串文字DOES的类型为array of char

因此,以上内容可以重写为:
char s1[]=" (),-089<>?BCFGHJLSVWZ[^hlmnxy|}";
char s2[]="                                !";
130+s1[s2[o8]-94];

您可以看到它是合法的C++。

10-02 14:45