Annotation of hatari/src/uae-cpu/readcpu.c, revision 1.1.1.1

1.1       root        1: /*
                      2:  * UAE - The Un*x Amiga Emulator
                      3:  *
                      4:  * Read 68000 CPU specs from file "table68k"
                      5:  *
                      6:  * Copyright 1995,1996 Bernd Schmidt
                      7:  */
                      8: 
                      9: #include <ctype.h>
                     10: /*
                     11: #include "sysconfig.h"
                     12: #include "config.h"
                     13: #include "options.h"
                     14: */
                     15: #include "sysdeps.h"
                     16: #include "readcpu.h"
                     17: 
                     18: int nr_cpuop_funcs;
                     19: 
                     20: struct mnemolookup lookuptab[] = {
                     21:     { i_ILLG, "ILLEGAL" },
                     22:     { i_OR, "OR" },
                     23:     { i_CHK, "CHK" },
                     24:     { i_CHK2, "CHK2" },
                     25:     { i_AND, "AND" },
                     26:     { i_EOR, "EOR" },
                     27:     { i_ORSR, "ORSR" },
                     28:     { i_ANDSR, "ANDSR" },
                     29:     { i_EORSR, "EORSR" },
                     30:     { i_SUB, "SUB" },
                     31:     { i_SUBA, "SUBA" },
                     32:     { i_SUBX, "SUBX" },
                     33:     { i_SBCD, "SBCD" },
                     34:     { i_ADD, "ADD" },
                     35:     { i_ADDA, "ADDA" },
                     36:     { i_ADDX, "ADDX" },
                     37:     { i_ABCD, "ABCD" },
                     38:     { i_NEG, "NEG" },
                     39:     { i_NEGX, "NEGX" },
                     40:     { i_NBCD, "NBCD" },
                     41:     { i_CLR, "CLR" },
                     42:     { i_NOT, "NOT" },
                     43:     { i_TST, "TST" },
                     44:     { i_BTST, "BTST" },
                     45:     { i_BCHG, "BCHG" },
                     46:     { i_BCLR, "BCLR" },
                     47:     { i_BSET, "BSET" },
                     48:     { i_CMP, "CMP" },
                     49:     { i_CMPM, "CMPM" },
                     50:     { i_CMPA, "CMPA" },
                     51:     { i_MVPRM, "MVPRM" },
                     52:     { i_MVPMR, "MVPMR" },
                     53:     { i_MOVE, "MOVE" },
                     54:     { i_MOVEA, "MOVEA" },
                     55:     { i_MVSR2, "MVSR2" },
                     56:     { i_MV2SR, "MV2SR" },
                     57:     { i_SWAP, "SWAP" },
                     58:     { i_EXG, "EXG" },
                     59:     { i_EXT, "EXT" },
                     60:     { i_MVMEL, "MVMEL" },
                     61:     { i_MVMLE, "MVMLE" },
                     62:     { i_TRAP, "TRAP" },
                     63:     { i_MVR2USP, "MVR2USP" },
                     64:     { i_MVUSP2R, "MVUSP2R" },
                     65:     { i_NOP, "NOP" },
                     66:     { i_RESET, "RESET" },
                     67:     { i_RTE, "RTE" },
                     68:     { i_RTD, "RTD" },
                     69:     { i_LINK, "LINK" },
                     70:     { i_UNLK, "UNLK" },
                     71:     { i_RTS, "RTS" },
                     72:     { i_STOP, "STOP" },
                     73:     { i_TRAPV, "TRAPV" },
                     74:     { i_RTR, "RTR" },
                     75:     { i_JSR, "JSR" },
                     76:     { i_JMP, "JMP" },
                     77:     { i_BSR, "BSR" },
                     78:     { i_Bcc, "Bcc" },
                     79:     { i_LEA, "LEA" },
                     80:     { i_PEA, "PEA" },
                     81:     { i_DBcc, "DBcc" },
                     82:     { i_Scc, "Scc" },
                     83:     { i_DIVU, "DIVU" },
                     84:     { i_DIVS, "DIVS" },
                     85:     { i_MULU, "MULU" },
                     86:     { i_MULS, "MULS" },
                     87:     { i_ASR, "ASR" },
                     88:     { i_ASL, "ASL" },
                     89:     { i_LSR, "LSR" },
                     90:     { i_LSL, "LSL" },
                     91:     { i_ROL, "ROL" },
                     92:     { i_ROR, "ROR" },
                     93:     { i_ROXL, "ROXL" },
                     94:     { i_ROXR, "ROXR" },
                     95:     { i_ASRW, "ASRW" },
                     96:     { i_ASLW, "ASLW" },
                     97:     { i_LSRW, "LSRW" },
                     98:     { i_LSLW, "LSLW" },
                     99:     { i_ROLW, "ROLW" },
                    100:     { i_RORW, "RORW" },
                    101:     { i_ROXLW, "ROXLW" },
                    102:     { i_ROXRW, "ROXRW" },
                    103: 
                    104:     { i_MOVE2C, "MOVE2C" },
                    105:     { i_MOVEC2, "MOVEC2" },
                    106:     { i_CAS, "CAS" },
                    107:     { i_CAS2, "CAS2" },
                    108:     { i_MULL, "MULL" },
                    109:     { i_DIVL, "DIVL" },
                    110:     { i_BFTST, "BFTST" },
                    111:     { i_BFEXTU, "BFEXTU" },
                    112:     { i_BFCHG, "BFCHG" },
                    113:     { i_BFEXTS, "BFEXTS" },
                    114:     { i_BFCLR, "BFCLR" },
                    115:     { i_BFFFO, "BFFFO" },
                    116:     { i_BFSET, "BFSET" },
                    117:     { i_BFINS, "BFINS" },
                    118:     { i_PACK, "PACK" },
                    119:     { i_UNPK, "UNPK" },
                    120:     { i_TAS, "TAS" },
                    121:     { i_BKPT, "BKPT" },
                    122:     { i_CALLM, "CALLM" },
                    123:     { i_RTM, "RTM" },
                    124:     { i_TRAPcc, "TRAPcc" },
                    125:     { i_MOVES, "MOVES" },
                    126:     { i_FPP, "FPP" },
                    127:     { i_FDBcc, "FDBcc" },
                    128:     { i_FScc, "FScc" },
                    129:     { i_FTRAPcc, "FTRAPcc" },
                    130:     { i_FBcc, "FBcc" },
                    131:     { i_FBcc, "FBcc" },
                    132:     { i_FSAVE, "FSAVE" },
                    133:     { i_FRESTORE, "FRESTORE" },
                    134: 
                    135:     { i_CINVL, "CINVL" },
                    136:     { i_CINVP, "CINVP" },
                    137:     { i_CINVA, "CINVA" },
                    138:     { i_CPUSHL, "CPUSHL" },
                    139:     { i_CPUSHP, "CPUSHP" },
                    140:     { i_CPUSHA, "CPUSHA" },
                    141:     { i_MOVE16, "MOVE16" },
                    142: 
                    143:     { i_MMUOP, "MMUOP" },
                    144:     { i_ILLG, "" },
                    145: };
                    146: 
                    147: struct instr *table68k;
                    148: 
                    149: STATIC_INLINE amodes mode_from_str (const char *str)
                    150: {
                    151:     if (strncmp (str, "Dreg", 4) == 0) return Dreg;
                    152:     if (strncmp (str, "Areg", 4) == 0) return Areg;
                    153:     if (strncmp (str, "Aind", 4) == 0) return Aind;
                    154:     if (strncmp (str, "Apdi", 4) == 0) return Apdi;
                    155:     if (strncmp (str, "Aipi", 4) == 0) return Aipi;
                    156:     if (strncmp (str, "Ad16", 4) == 0) return Ad16;
                    157:     if (strncmp (str, "Ad8r", 4) == 0) return Ad8r;
                    158:     if (strncmp (str, "absw", 4) == 0) return absw;
                    159:     if (strncmp (str, "absl", 4) == 0) return absl;
                    160:     if (strncmp (str, "PC16", 4) == 0) return PC16;
                    161:     if (strncmp (str, "PC8r", 4) == 0) return PC8r;
                    162:     if (strncmp (str, "Immd", 4) == 0) return imm;
                    163:     abort ();
                    164:     return 0;
                    165: }
                    166: 
                    167: STATIC_INLINE amodes mode_from_mr (int mode, int reg)
                    168: {
                    169:     switch (mode) {
                    170:      case 0: return Dreg;
                    171:      case 1: return Areg;
                    172:      case 2: return Aind;
                    173:      case 3: return Aipi;
                    174:      case 4: return Apdi;
                    175:      case 5: return Ad16;
                    176:      case 6: return Ad8r;
                    177:      case 7:
                    178:        switch (reg) {
                    179:         case 0: return absw;
                    180:         case 1: return absl;
                    181:         case 2: return PC16;
                    182:         case 3: return PC8r;
                    183:         case 4: return imm;
                    184:         case 5:
                    185:         case 6:
                    186:         case 7: return am_illg;
                    187:        }
                    188:     }
                    189:     abort ();
                    190:     return 0;
                    191: }
                    192: 
                    193: static void build_insn (int insn)
                    194: {
                    195:     int find = -1;
                    196:     int variants;
                    197:     struct instr_def id;
                    198:     const char *opcstr;
                    199:     int i;
                    200: 
                    201:     int flaglive = 0, flagdead = 0;
                    202: 
                    203:     id = defs68k[insn];
                    204: 
                    205:     for (i = 0; i < 5; i++) {
                    206:        switch (id.flaginfo[i].flagset){
                    207:         case fa_unset: break;
                    208:         case fa_isjmp: break;
                    209:         case fa_zero: flagdead |= 1 << i; break;
                    210:         case fa_one: flagdead |= 1 << i; break;
                    211:         case fa_dontcare: flagdead |= 1 << i; break;
                    212:         case fa_unknown: flagdead = -1; goto out1;
                    213:         case fa_set: flagdead |= 1 << i; break;
                    214:        }
                    215:     }
                    216: 
                    217:     out1:
                    218:     for (i = 0; i < 5; i++) {
                    219:        switch (id.flaginfo[i].flaguse) {
                    220:         case fu_unused: break;
                    221:         case fu_isjmp: flaglive |= 1 << i; break;
                    222:         case fu_maybecc: flaglive |= 1 << i; break;
                    223:         case fu_unknown: flaglive = -1; goto out2;
                    224:         case fu_used: flaglive |= 1 << i; break;
                    225:        }
                    226:     }
                    227:     out2:
                    228: 
                    229:     opcstr = id.opcstr;
                    230:     for (variants = 0; variants < (1 << id.n_variable); variants++) {
                    231:        int bitcnt[lastbit];
                    232:        int bitval[lastbit];
                    233:        int bitpos[lastbit];
                    234:        int i;
                    235:        uae_u16 opc = id.bits;
                    236:        uae_u16 msk, vmsk;
                    237:        int pos = 0;
                    238:        int mnp = 0;
                    239:        int bitno = 0;
                    240:        char mnemonic[10];
                    241: 
                    242:        wordsizes sz = sz_long;
                    243:        int srcgather = 0, dstgather = 0;
                    244:        int usesrc = 0, usedst = 0;
                    245:        int srctype = 0;
                    246:        int srcpos = -1, dstpos = -1;
                    247: 
                    248:        amodes srcmode = am_unknown, destmode = am_unknown;
                    249:        int srcreg = -1, destreg = -1;
                    250: 
                    251:        for (i = 0; i < lastbit; i++)
                    252:            bitcnt[i] = bitval[i] = 0;
                    253: 
                    254:        vmsk = 1 << id.n_variable;
                    255: 
                    256:        for (i = 0, msk = 0x8000; i < 16; i++, msk >>= 1) {
                    257:            if (!(msk & id.mask)) {
                    258:                int currbit = id.bitpos[bitno++];
                    259:                int bit_set;
                    260:                vmsk >>= 1;
                    261:                bit_set = variants & vmsk ? 1 : 0;
                    262:                if (bit_set)
                    263:                    opc |= msk;
                    264:                bitpos[currbit] = 15 - i;
                    265:                bitcnt[currbit]++;
                    266:                bitval[currbit] <<= 1;
                    267:                bitval[currbit] |= bit_set;
                    268:            }
                    269:        }
                    270: 
                    271:        if (bitval[bitj] == 0) bitval[bitj] = 8;
                    272:        /* first check whether this one does not match after all */
                    273:        if (bitval[bitz] == 3 || bitval[bitC] == 1)
                    274:            continue;
                    275:        if (bitcnt[bitI] && (bitval[bitI] == 0x00 || bitval[bitI] == 0xff))
                    276:            continue;
                    277: 
                    278:        /* bitI and bitC get copied to biti and bitc */
                    279:        if (bitcnt[bitI]) {
                    280:            bitval[biti] = bitval[bitI]; bitpos[biti] = bitpos[bitI];
                    281:        }
                    282:        if (bitcnt[bitC])
                    283:            bitval[bitc] = bitval[bitC];
                    284: 
                    285:        pos = 0;
                    286:        while (opcstr[pos] && !isspace(opcstr[pos])) {
                    287:            if (opcstr[pos] == '.') {
                    288:                pos++;
                    289:                switch (opcstr[pos]) {
                    290: 
                    291:                 case 'B': sz = sz_byte; break;
                    292:                 case 'W': sz = sz_word; break;
                    293:                 case 'L': sz = sz_long; break;
                    294:                 case 'z':
                    295:                    switch (bitval[bitz]) {
                    296:                     case 0: sz = sz_byte; break;
                    297:                     case 1: sz = sz_word; break;
                    298:                     case 2: sz = sz_long; break;
                    299:                     default: abort();
                    300:                    }
                    301:                    break;
                    302:                 default: abort();
                    303:                }
                    304:            } else {
                    305:                mnemonic[mnp] = opcstr[pos];
                    306:                if (mnemonic[mnp] == 'f') {
                    307:                    find = -1;
                    308:                    switch (bitval[bitf]) {
                    309:                     case 0: mnemonic[mnp] = 'R'; break;
                    310:                     case 1: mnemonic[mnp] = 'L'; break;
                    311:                     default: abort();
                    312:                    }
                    313:                }
                    314:                mnp++;
                    315:            }
                    316:            pos++;
                    317:        }
                    318:        mnemonic[mnp] = 0;
                    319: 
                    320:        /* now, we have read the mnemonic and the size */
                    321:        while (opcstr[pos] && isspace(opcstr[pos]))
                    322:            pos++;
                    323: 
                    324:        /* A goto a day keeps the D******a away. */
                    325:        if (opcstr[pos] == 0)
                    326:            goto endofline;
                    327: 
                    328:        /* parse the source address */
                    329:        usesrc = 1;
                    330:        switch (opcstr[pos++]) {
                    331:         case 'D':
                    332:            srcmode = Dreg;
                    333:            switch (opcstr[pos++]) {
                    334:             case 'r': srcreg = bitval[bitr]; srcgather = 1; srcpos = bitpos[bitr]; break;
                    335:             case 'R': srcreg = bitval[bitR]; srcgather = 1; srcpos = bitpos[bitR]; break;
                    336:             default: abort();
                    337:            }
                    338: 
                    339:            break;
                    340:         case 'A':
                    341:            srcmode = Areg;
                    342:            switch (opcstr[pos++]) {
                    343:             case 'r': srcreg = bitval[bitr]; srcgather = 1; srcpos = bitpos[bitr]; break;
                    344:             case 'R': srcreg = bitval[bitR]; srcgather = 1; srcpos = bitpos[bitR]; break;
                    345:             default: abort();
                    346:            }
                    347:            switch (opcstr[pos]) {
                    348:             case 'p': srcmode = Apdi; pos++; break;
                    349:             case 'P': srcmode = Aipi; pos++; break;
                    350:            }
                    351:            break;
                    352:         case '#':
                    353:            switch (opcstr[pos++]) {
                    354:             case 'z': srcmode = imm; break;
                    355:             case '0': srcmode = imm0; break;
                    356:             case '1': srcmode = imm1; break;
                    357:             case '2': srcmode = imm2; break;
                    358:             case 'i': srcmode = immi; srcreg = (uae_s32)(uae_s8)bitval[biti];
                    359:                if (CPU_EMU_SIZE < 4) {
                    360:                    /* Used for branch instructions */
                    361:                    srctype = 1;
                    362:                    srcgather = 1;
                    363:                    srcpos = bitpos[biti];
                    364:                }
                    365:                break;
                    366:             case 'j': srcmode = immi; srcreg = bitval[bitj];
                    367:                if (CPU_EMU_SIZE < 3) {
                    368:                    /* 1..8 for ADDQ/SUBQ and rotshi insns */
                    369:                    srcgather = 1;
                    370:                    srctype = 3;
                    371:                    srcpos = bitpos[bitj];
                    372:                }
                    373:                break;
                    374:             case 'J': srcmode = immi; srcreg = bitval[bitJ];
                    375:                if (CPU_EMU_SIZE < 5) {
                    376:                    /* 0..15 */
                    377:                    srcgather = 1;
                    378:                    srctype = 2;
                    379:                    srcpos = bitpos[bitJ];
                    380:                }
                    381:                break;
                    382:             case 'k': srcmode = immi; srcreg = bitval[bitk];
                    383:                if (CPU_EMU_SIZE < 3) {
                    384:                    srcgather = 1;
                    385:                    srctype = 4;
                    386:                    srcpos = bitpos[bitk];
                    387:                }
                    388:                break;
                    389:             case 'K': srcmode = immi; srcreg = bitval[bitK];
                    390:                if (CPU_EMU_SIZE < 5) {
                    391:                    /* 0..15 */
                    392:                    srcgather = 1;
                    393:                    srctype = 5;
                    394:                    srcpos = bitpos[bitK];
                    395:                }
                    396:                break;
                    397:             default: abort();
                    398:            }
                    399:            break;
                    400:         case 'd':
                    401:            srcreg = bitval[bitD];
                    402:            srcmode = mode_from_mr(bitval[bitd],bitval[bitD]);
                    403:            if (srcmode == am_illg)
                    404:                continue;
                    405:            if (CPU_EMU_SIZE < 2 &&
                    406:                (srcmode == Areg || srcmode == Dreg || srcmode == Aind
                    407:                 || srcmode == Ad16 || srcmode == Ad8r || srcmode == Aipi
                    408:                 || srcmode == Apdi))
                    409:            {
                    410:                srcgather = 1; srcpos = bitpos[bitD];
                    411:            }
                    412:            if (opcstr[pos] == '[') {
                    413:                pos++;
                    414:                if (opcstr[pos] == '!') {
                    415:                    /* exclusion */
                    416:                    do {
                    417:                        pos++;
                    418:                        if (mode_from_str(opcstr+pos) == srcmode)
                    419:                            goto nomatch;
                    420:                        pos += 4;
                    421:                    } while (opcstr[pos] == ',');
                    422:                    pos++;
                    423:                } else {
                    424:                    if (opcstr[pos+4] == '-') {
                    425:                        /* replacement */
                    426:                        if (mode_from_str(opcstr+pos) == srcmode)
                    427:                            srcmode = mode_from_str(opcstr+pos+5);
                    428:                        else
                    429:                            goto nomatch;
                    430:                        pos += 10;
                    431:                    } else {
                    432:                        /* normal */
                    433:                        while(mode_from_str(opcstr+pos) != srcmode) {
                    434:                            pos += 4;
                    435:                            if (opcstr[pos] == ']')
                    436:                                goto nomatch;
                    437:                            pos++;
                    438:                        }
                    439:                        while(opcstr[pos] != ']') pos++;
                    440:                        pos++;
                    441:                        break;
                    442:                    }
                    443:                }
                    444:            }
                    445:            /* Some addressing modes are invalid as destination */
                    446:            if (srcmode == imm || srcmode == PC16 || srcmode == PC8r)
                    447:                goto nomatch;
                    448:            break;
                    449:         case 's':
                    450:            srcreg = bitval[bitS];
                    451:            srcmode = mode_from_mr(bitval[bits],bitval[bitS]);
                    452: 
                    453:            if (srcmode == am_illg)
                    454:                continue;
                    455:            if (CPU_EMU_SIZE < 2 &&
                    456:                (srcmode == Areg || srcmode == Dreg || srcmode == Aind
                    457:                 || srcmode == Ad16 || srcmode == Ad8r || srcmode == Aipi
                    458:                 || srcmode == Apdi))
                    459:            {
                    460:                srcgather = 1; srcpos = bitpos[bitS];
                    461:            }
                    462:            if (opcstr[pos] == '[') {
                    463:                pos++;
                    464:                if (opcstr[pos] == '!') {
                    465:                    /* exclusion */
                    466:                    do {
                    467:                        pos++;
                    468:                        if (mode_from_str(opcstr+pos) == srcmode)
                    469:                            goto nomatch;
                    470:                        pos += 4;
                    471:                    } while (opcstr[pos] == ',');
                    472:                    pos++;
                    473:                } else {
                    474:                    if (opcstr[pos+4] == '-') {
                    475:                        /* replacement */
                    476:                        if (mode_from_str(opcstr+pos) == srcmode)
                    477:                            srcmode = mode_from_str(opcstr+pos+5);
                    478:                        else
                    479:                            goto nomatch;
                    480:                        pos += 10;
                    481:                    } else {
                    482:                        /* normal */
                    483:                        while(mode_from_str(opcstr+pos) != srcmode) {
                    484:                            pos += 4;
                    485:                            if (opcstr[pos] == ']')
                    486:                                goto nomatch;
                    487:                            pos++;
                    488:                        }
                    489:                        while(opcstr[pos] != ']') pos++;
                    490:                        pos++;
                    491:                    }
                    492:                }
                    493:            }
                    494:            break;
                    495:         default: abort();
                    496:        }
                    497:        /* safety check - might have changed */
                    498:        if (srcmode != Areg && srcmode != Dreg && srcmode != Aind
                    499:            && srcmode != Ad16 && srcmode != Ad8r && srcmode != Aipi
                    500:            && srcmode != Apdi && srcmode != immi)
                    501:        {
                    502:            srcgather = 0;
                    503:        }
                    504:        if (srcmode == Areg && sz == sz_byte)
                    505:            goto nomatch;
                    506: 
                    507:        if (opcstr[pos] != ',')
                    508:            goto endofline;
                    509:        pos++;
                    510: 
                    511:        /* parse the destination address */
                    512:        usedst = 1;
                    513:        switch (opcstr[pos++]) {
                    514:         case 'D':
                    515:            destmode = Dreg;
                    516:            switch (opcstr[pos++]) {
                    517:             case 'r': destreg = bitval[bitr]; dstgather = 1; dstpos = bitpos[bitr]; break;
                    518:             case 'R': destreg = bitval[bitR]; dstgather = 1; dstpos = bitpos[bitR]; break;
                    519:             default: abort();
                    520:            }
                    521:            break;
                    522:         case 'A':
                    523:            destmode = Areg;
                    524:            switch (opcstr[pos++]) {
                    525:             case 'r': destreg = bitval[bitr]; dstgather = 1; dstpos = bitpos[bitr]; break;
                    526:             case 'R': destreg = bitval[bitR]; dstgather = 1; dstpos = bitpos[bitR]; break;
                    527:             default: abort();
                    528:            }
                    529:            switch (opcstr[pos]) {
                    530:             case 'p': destmode = Apdi; pos++; break;
                    531:             case 'P': destmode = Aipi; pos++; break;
                    532:            }
                    533:            break;
                    534:         case '#':
                    535:            switch (opcstr[pos++]) {
                    536:             case 'z': destmode = imm; break;
                    537:             case '0': destmode = imm0; break;
                    538:             case '1': destmode = imm1; break;
                    539:             case '2': destmode = imm2; break;
                    540:             case 'i': destmode = immi; destreg = (uae_s32)(uae_s8)bitval[biti]; break;
                    541:             case 'j': destmode = immi; destreg = bitval[bitj]; break;
                    542:             case 'J': destmode = immi; destreg = bitval[bitJ]; break;
                    543:             case 'k': destmode = immi; destreg = bitval[bitk]; break;
                    544:             case 'K': destmode = immi; destreg = bitval[bitK]; break;
                    545:             default: abort();
                    546:            }
                    547:            break;
                    548:         case 'd':
                    549:            destreg = bitval[bitD];
                    550:            destmode = mode_from_mr(bitval[bitd],bitval[bitD]);
                    551:            if (destmode == am_illg)
                    552:                continue;
                    553:            if (CPU_EMU_SIZE < 1 &&
                    554:                (destmode == Areg || destmode == Dreg || destmode == Aind
                    555:                 || destmode == Ad16 || destmode == Ad8r || destmode == Aipi
                    556:                 || destmode == Apdi))
                    557:            {
                    558:                dstgather = 1; dstpos = bitpos[bitD];
                    559:            }
                    560: 
                    561:            if (opcstr[pos] == '[') {
                    562:                pos++;
                    563:                if (opcstr[pos] == '!') {
                    564:                    /* exclusion */
                    565:                    do {
                    566:                        pos++;
                    567:                        if (mode_from_str(opcstr+pos) == destmode)
                    568:                            goto nomatch;
                    569:                        pos += 4;
                    570:                    } while (opcstr[pos] == ',');
                    571:                    pos++;
                    572:                } else {
                    573:                    if (opcstr[pos+4] == '-') {
                    574:                        /* replacement */
                    575:                        if (mode_from_str(opcstr+pos) == destmode)
                    576:                            destmode = mode_from_str(opcstr+pos+5);
                    577:                        else
                    578:                            goto nomatch;
                    579:                        pos += 10;
                    580:                    } else {
                    581:                        /* normal */
                    582:                        while(mode_from_str(opcstr+pos) != destmode) {
                    583:                            pos += 4;
                    584:                            if (opcstr[pos] == ']')
                    585:                                goto nomatch;
                    586:                            pos++;
                    587:                        }
                    588:                        while(opcstr[pos] != ']') pos++;
                    589:                        pos++;
                    590:                        break;
                    591:                    }
                    592:                }
                    593:            }
                    594:            /* Some addressing modes are invalid as destination */
                    595:            if (destmode == imm || destmode == PC16 || destmode == PC8r)
                    596:                goto nomatch;
                    597:            break;
                    598:         case 's':
                    599:            destreg = bitval[bitS];
                    600:            destmode = mode_from_mr(bitval[bits],bitval[bitS]);
                    601: 
                    602:            if (destmode == am_illg)
                    603:                continue;
                    604:            if (CPU_EMU_SIZE < 1 &&
                    605:                (destmode == Areg || destmode == Dreg || destmode == Aind
                    606:                 || destmode == Ad16 || destmode == Ad8r || destmode == Aipi
                    607:                 || destmode == Apdi))
                    608:            {
                    609:                dstgather = 1; dstpos = bitpos[bitS];
                    610:            }
                    611: 
                    612:            if (opcstr[pos] == '[') {
                    613:                pos++;
                    614:                if (opcstr[pos] == '!') {
                    615:                    /* exclusion */
                    616:                    do {
                    617:                        pos++;
                    618:                        if (mode_from_str(opcstr+pos) == destmode)
                    619:                            goto nomatch;
                    620:                        pos += 4;
                    621:                    } while (opcstr[pos] == ',');
                    622:                    pos++;
                    623:                } else {
                    624:                    if (opcstr[pos+4] == '-') {
                    625:                        /* replacement */
                    626:                        if (mode_from_str(opcstr+pos) == destmode)
                    627:                            destmode = mode_from_str(opcstr+pos+5);
                    628:                        else
                    629:                            goto nomatch;
                    630:                        pos += 10;
                    631:                    } else {
                    632:                        /* normal */
                    633:                        while(mode_from_str(opcstr+pos) != destmode) {
                    634:                            pos += 4;
                    635:                            if (opcstr[pos] == ']')
                    636:                                goto nomatch;
                    637:                            pos++;
                    638:                        }
                    639:                        while(opcstr[pos] != ']') pos++;
                    640:                        pos++;
                    641:                    }
                    642:                }
                    643:            }
                    644:            break;
                    645:         default: abort();
                    646:        }
                    647:        /* safety check - might have changed */
                    648:        if (destmode != Areg && destmode != Dreg && destmode != Aind
                    649:            && destmode != Ad16 && destmode != Ad8r && destmode != Aipi
                    650:            && destmode != Apdi)
                    651:        {
                    652:            dstgather = 0;
                    653:        }
                    654: 
                    655:        if (destmode == Areg && sz == sz_byte)
                    656:            goto nomatch;
                    657: #if 0
                    658:        if (sz == sz_byte && (destmode == Aipi || destmode == Apdi)) {
                    659:            dstgather = 0;
                    660:        }
                    661: #endif
                    662:        endofline:
                    663:        /* now, we have a match */
                    664:        if (table68k[opc].mnemo != i_ILLG)
                    665:            fprintf(stderr, "Double match: %x: %s\n", opc, opcstr);
                    666:        if (find == -1) {
                    667:            for (find = 0;; find++) {
                    668:                if (strcmp(mnemonic, lookuptab[find].name) == 0) {
                    669:                    table68k[opc].mnemo = lookuptab[find].mnemo;
                    670:                    break;
                    671:                }
                    672:                if (strlen(lookuptab[find].name) == 0) abort();
                    673:            }
                    674:        }
                    675:        else {
                    676:            table68k[opc].mnemo = lookuptab[find].mnemo;
                    677:        }
                    678:        table68k[opc].cc = bitval[bitc];
                    679:        if (table68k[opc].mnemo == i_BTST
                    680:            || table68k[opc].mnemo == i_BSET
                    681:            || table68k[opc].mnemo == i_BCLR
                    682:            || table68k[opc].mnemo == i_BCHG)
                    683:        {
                    684:            sz = destmode == Dreg ? sz_long : sz_byte;
                    685:        }
                    686:        table68k[opc].size = sz;
                    687:        table68k[opc].sreg = srcreg;
                    688:        table68k[opc].dreg = destreg;
                    689:        table68k[opc].smode = srcmode;
                    690:        table68k[opc].dmode = destmode;
                    691:        table68k[opc].spos = srcgather ? srcpos : -1;
                    692:        table68k[opc].dpos = dstgather ? dstpos : -1;
                    693:        table68k[opc].suse = usesrc;
                    694:        table68k[opc].duse = usedst;
                    695:        table68k[opc].stype = srctype;
                    696:        table68k[opc].plev = id.plevel;
                    697:        table68k[opc].clev = id.cpulevel;
                    698: #if 0
                    699:        for (i = 0; i < 5; i++) {
                    700:            table68k[opc].flaginfo[i].flagset = id.flaginfo[i].flagset;
                    701:            table68k[opc].flaginfo[i].flaguse = id.flaginfo[i].flaguse;
                    702:        }
                    703: #endif
                    704:        table68k[opc].flagdead = flagdead;
                    705:        table68k[opc].flaglive = flaglive;
                    706:        nomatch:
                    707:        /* FOO! */;
                    708:     }
                    709: }
                    710: 
                    711: 
                    712: void read_table68k (void)
                    713: {
                    714:     int i;
                    715: 
                    716:     table68k = (struct instr *)malloc (65536 * sizeof (struct instr));
                    717:     for (i = 0; i < 65536; i++) {
                    718:        table68k[i].mnemo = i_ILLG;
                    719:        table68k[i].handler = -1;
                    720:     }
                    721:     for (i = 0; i < n_defs68k; i++) {
                    722:        build_insn (i);
                    723:     }
                    724: }
                    725: 
                    726: static int mismatch;
                    727: 
                    728: static void handle_merges (long int opcode)
                    729: {
                    730:     uae_u16 smsk;
                    731:     uae_u16 dmsk;
                    732:     int sbitdst, dstend;
                    733:     int srcreg, dstreg;
                    734: 
                    735:     if (table68k[opcode].spos == -1) {
                    736:        sbitdst = 1; smsk = 0;
                    737:     } else {
                    738:        switch (table68k[opcode].stype) {
                    739:         case 0:
                    740:            smsk = 7; sbitdst = 8; break;
                    741:         case 1:
                    742:            smsk = 255; sbitdst = 256; break;
                    743:         case 2:
                    744:            smsk = 15; sbitdst = 16; break;
                    745:         case 3:
                    746:            smsk = 7; sbitdst = 8; break;
                    747:         case 4:
                    748:            smsk = 7; sbitdst = 8; break;
                    749:         case 5:
                    750:            smsk = 63; sbitdst = 64; break;
                    751:         default:
                    752:            smsk = 0; sbitdst = 0;
                    753:            abort();
                    754:            break;
                    755:        }
                    756:        smsk <<= table68k[opcode].spos;
                    757:     }
                    758:     if (table68k[opcode].dpos == -1) {
                    759:        dstend = 1; dmsk = 0;
                    760:     } else {
                    761:        dmsk = 7 << table68k[opcode].dpos;
                    762:        dstend = 8;
                    763:     }
                    764:     for (srcreg=0; srcreg < sbitdst; srcreg++) {
                    765:        for (dstreg=0; dstreg < dstend; dstreg++) {
                    766:            uae_u16 code = opcode;
                    767: 
                    768:            code = (code & ~smsk) | (srcreg << table68k[opcode].spos);
                    769:            code = (code & ~dmsk) | (dstreg << table68k[opcode].dpos);
                    770: 
                    771:            /* Check whether this is in fact the same instruction.
                    772:             * The instructions should never differ, except for the
                    773:             * Bcc.(BW) case. */
                    774:            if (table68k[code].mnemo != table68k[opcode].mnemo
                    775:                || table68k[code].size != table68k[opcode].size
                    776:                || table68k[code].suse != table68k[opcode].suse
                    777:                || table68k[code].duse != table68k[opcode].duse)
                    778:            {
                    779:                mismatch++; continue;
                    780:            }
                    781:            if (table68k[opcode].suse
                    782:                && (table68k[opcode].spos != table68k[code].spos
                    783:                    || table68k[opcode].smode != table68k[code].smode
                    784:                    || table68k[opcode].stype != table68k[code].stype))
                    785:            {
                    786:                mismatch++; continue;
                    787:            }
                    788:            if (table68k[opcode].duse
                    789:                && (table68k[opcode].dpos != table68k[code].dpos
                    790:                    || table68k[opcode].dmode != table68k[code].dmode))
                    791:            {
                    792:                mismatch++; continue;
                    793:            }
                    794: 
                    795:            if (code != opcode)
                    796:                table68k[code].handler = opcode;
                    797:        }
                    798:     }
                    799: }
                    800: 
                    801: void do_merges (void)
                    802: {
                    803:     long int opcode;
                    804:     int nr = 0;
                    805:     mismatch = 0;
                    806:     for (opcode = 0; opcode < 65536; opcode++) {
                    807:        if (table68k[opcode].handler != -1 || table68k[opcode].mnemo == i_ILLG)
                    808:            continue;
                    809:        nr++;
                    810:        handle_merges (opcode);
                    811:     }
                    812:     nr_cpuop_funcs = nr;
                    813: }
                    814: 
                    815: int get_no_mismatches (void)
                    816: {
                    817:     return mismatch;
                    818: }

unix.superglobalmegacorp.com

This archive runs on limited infrastructure. Preserving old code on modern bandwidth. Automated agents are requested to crawl responsibly.