| /* |
| * UAE - The Un*x Amiga Emulator |
| * |
| * Read 68000 CPU specs from file "table68k" |
| * |
| * Copyright 1995,1996 Bernd Schmidt |
| * |
| * This program is free software; you can redistribute it and/or modify |
| * it under the terms of the GNU General Public License as published by |
| * the Free Software Foundation; either version 2 of the License, or |
| * (at your option) any later version. |
| * |
| * This program is distributed in the hope that it will be useful, |
| * but WITHOUT ANY WARRANTY; without even the implied warranty of |
| * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the |
| * GNU General Public License for more details. |
| * |
| * You should have received a copy of the GNU General Public License |
| * along with this program; if not, write to the Free Software |
| * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA |
| */ |
| |
| #include <stdio.h> |
| #include <stdlib.h> |
| #include <string.h> |
| #include <ctype.h> |
| |
| #include "sysdeps.h" |
| #include "readcpu.h" |
| |
| int nr_cpuop_funcs; |
| |
| struct mnemolookup lookuptab[] = { |
| { i_ILLG, "ILLEGAL" }, |
| { i_OR, "OR" }, |
| { i_CHK, "CHK" }, |
| { i_CHK2, "CHK2" }, |
| { i_AND, "AND" }, |
| { i_EOR, "EOR" }, |
| { i_ORSR, "ORSR" }, |
| { i_ANDSR, "ANDSR" }, |
| { i_EORSR, "EORSR" }, |
| { i_SUB, "SUB" }, |
| { i_SUBA, "SUBA" }, |
| { i_SUBX, "SUBX" }, |
| { i_SBCD, "SBCD" }, |
| { i_ADD, "ADD" }, |
| { i_ADDA, "ADDA" }, |
| { i_ADDX, "ADDX" }, |
| { i_ABCD, "ABCD" }, |
| { i_NEG, "NEG" }, |
| { i_NEGX, "NEGX" }, |
| { i_NBCD, "NBCD" }, |
| { i_CLR, "CLR" }, |
| { i_NOT, "NOT" }, |
| { i_TST, "TST" }, |
| { i_BTST, "BTST" }, |
| { i_BCHG, "BCHG" }, |
| { i_BCLR, "BCLR" }, |
| { i_BSET, "BSET" }, |
| { i_CMP, "CMP" }, |
| { i_CMPM, "CMPM" }, |
| { i_CMPA, "CMPA" }, |
| { i_MVPRM, "MVPRM" }, |
| { i_MVPMR, "MVPMR" }, |
| { i_MOVE, "MOVE" }, |
| { i_MOVEA, "MOVEA" }, |
| { i_MVSR2, "MVSR2" }, |
| { i_MV2SR, "MV2SR" }, |
| { i_SWAP, "SWAP" }, |
| { i_EXG, "EXG" }, |
| { i_EXT, "EXT" }, |
| { i_MVMEL, "MVMEL" }, |
| { i_MVMLE, "MVMLE" }, |
| { i_TRAP, "TRAP" }, |
| { i_MVR2USP, "MVR2USP" }, |
| { i_MVUSP2R, "MVUSP2R" }, |
| { i_NOP, "NOP" }, |
| { i_RESET, "RESET" }, |
| { i_RTE, "RTE" }, |
| { i_RTD, "RTD" }, |
| { i_LINK, "LINK" }, |
| { i_UNLK, "UNLK" }, |
| { i_RTS, "RTS" }, |
| { i_STOP, "STOP" }, |
| { i_TRAPV, "TRAPV" }, |
| { i_RTR, "RTR" }, |
| { i_JSR, "JSR" }, |
| { i_JMP, "JMP" }, |
| { i_BSR, "BSR" }, |
| { i_Bcc, "Bcc" }, |
| { i_LEA, "LEA" }, |
| { i_PEA, "PEA" }, |
| { i_DBcc, "DBcc" }, |
| { i_Scc, "Scc" }, |
| { i_DIVU, "DIVU" }, |
| { i_DIVS, "DIVS" }, |
| { i_MULU, "MULU" }, |
| { i_MULS, "MULS" }, |
| { i_ASR, "ASR" }, |
| { i_ASL, "ASL" }, |
| { i_LSR, "LSR" }, |
| { i_LSL, "LSL" }, |
| { i_ROL, "ROL" }, |
| { i_ROR, "ROR" }, |
| { i_ROXL, "ROXL" }, |
| { i_ROXR, "ROXR" }, |
| { i_ASRW, "ASRW" }, |
| { i_ASLW, "ASLW" }, |
| { i_LSRW, "LSRW" }, |
| { i_LSLW, "LSLW" }, |
| { i_ROLW, "ROLW" }, |
| { i_RORW, "RORW" }, |
| { i_ROXLW, "ROXLW" }, |
| { i_ROXRW, "ROXRW" }, |
| |
| { i_MOVE2C, "MOVE2C" }, |
| { i_MOVEC2, "MOVEC2" }, |
| { i_CAS, "CAS" }, |
| { i_CAS2, "CAS2" }, |
| { i_MULL, "MULL" }, |
| { i_DIVL, "DIVL" }, |
| { i_BFTST, "BFTST" }, |
| { i_BFEXTU, "BFEXTU" }, |
| { i_BFCHG, "BFCHG" }, |
| { i_BFEXTS, "BFEXTS" }, |
| { i_BFCLR, "BFCLR" }, |
| { i_BFFFO, "BFFFO" }, |
| { i_BFSET, "BFSET" }, |
| { i_BFINS, "BFINS" }, |
| { i_PACK, "PACK" }, |
| { i_UNPK, "UNPK" }, |
| { i_TAS, "TAS" }, |
| { i_BKPT, "BKPT" }, |
| { i_CALLM, "CALLM" }, |
| { i_RTM, "RTM" }, |
| { i_TRAPcc, "TRAPcc" }, |
| { i_MOVES, "MOVES" }, |
| { i_FPP, "FPP" }, |
| { i_FDBcc, "FDBcc" }, |
| { i_FScc, "FScc" }, |
| { i_FTRAPcc, "FTRAPcc" }, |
| { i_FBcc, "FBcc" }, |
| { i_FBcc, "FBcc" }, |
| { i_FSAVE, "FSAVE" }, |
| { i_FRESTORE, "FRESTORE" }, |
| |
| { i_CINVL, "CINVL" }, |
| { i_CINVP, "CINVP" }, |
| { i_CINVA, "CINVA" }, |
| { i_CPUSHL, "CPUSHL" }, |
| { i_CPUSHP, "CPUSHP" }, |
| { i_CPUSHA, "CPUSHA" }, |
| { i_MOVE16, "MOVE16" }, |
| |
| { i_EMULOP_RETURN, "EMULOP_RETURN" }, |
| { i_EMULOP, "EMULOP" }, |
| |
| { i_MMUOP, "MMUOP" }, |
| { i_ILLG, "" }, |
| }; |
| |
| struct instr *table68k; |
| |
| static __inline__ amodes mode_from_str (const char *str) |
| { |
| if (strncmp (str, "Dreg", 4) == 0) return Dreg; |
| if (strncmp (str, "Areg", 4) == 0) return Areg; |
| if (strncmp (str, "Aind", 4) == 0) return Aind; |
| if (strncmp (str, "Apdi", 4) == 0) return Apdi; |
| if (strncmp (str, "Aipi", 4) == 0) return Aipi; |
| if (strncmp (str, "Ad16", 4) == 0) return Ad16; |
| if (strncmp (str, "Ad8r", 4) == 0) return Ad8r; |
| if (strncmp (str, "absw", 4) == 0) return absw; |
| if (strncmp (str, "absl", 4) == 0) return absl; |
| if (strncmp (str, "PC16", 4) == 0) return PC16; |
| if (strncmp (str, "PC8r", 4) == 0) return PC8r; |
| if (strncmp (str, "Immd", 4) == 0) return imm; |
| abort (); |
| return (amodes)0; |
| } |
| |
| static __inline__ amodes mode_from_mr (int mode, int reg) |
| { |
| switch (mode) { |
| case 0: return Dreg; |
| case 1: return Areg; |
| case 2: return Aind; |
| case 3: return Aipi; |
| case 4: return Apdi; |
| case 5: return Ad16; |
| case 6: return Ad8r; |
| case 7: |
| switch (reg) { |
| case 0: return absw; |
| case 1: return absl; |
| case 2: return PC16; |
| case 3: return PC8r; |
| case 4: return imm; |
| case 5: |
| case 6: |
| case 7: return am_illg; |
| } |
| } |
| abort (); |
| return (amodes)0; |
| } |
| |
| static void build_insn (int insn) |
| { |
| int find = -1; |
| int variants; |
| struct instr_def id; |
| const char *opcstr; |
| int i, n; |
| |
| int flaglive = 0, flagdead = 0; |
| int cflow = 0; |
| |
| id = defs68k[insn]; |
| |
| // Control flow information |
| cflow = id.cflow; |
| |
| // Mask of flags set/used |
| unsigned char flags_set(0), flags_used(0); |
| |
| for (i = 0, n = 4; i < 5; i++, n--) { |
| switch (id.flaginfo[i].flagset) { |
| case fa_unset: case fa_isjmp: break; |
| default: flags_set |= (1 << n); |
| } |
| |
| switch (id.flaginfo[i].flaguse) { |
| case fu_unused: case fu_isjmp: break; |
| default: flags_used |= (1 << n); |
| } |
| } |
| |
| for (i = 0; i < 5; i++) { |
| switch (id.flaginfo[i].flagset){ |
| case fa_unset: break; |
| case fa_zero: flagdead |= 1 << i; break; |
| case fa_one: flagdead |= 1 << i; break; |
| case fa_dontcare: flagdead |= 1 << i; break; |
| case fa_unknown: flagdead = -1; goto out1; |
| case fa_set: flagdead |= 1 << i; break; |
| } |
| } |
| |
| out1: |
| for (i = 0; i < 5; i++) { |
| switch (id.flaginfo[i].flaguse) { |
| case fu_unused: break; |
| case fu_unknown: flaglive = -1; goto out2; |
| case fu_used: flaglive |= 1 << i; break; |
| } |
| } |
| out2: |
| |
| opcstr = id.opcstr; |
| for (variants = 0; variants < (1 << id.n_variable); variants++) { |
| int bitcnt[lastbit]; |
| int bitval[lastbit]; |
| int bitpos[lastbit]; |
| int i; |
| uae_u16 opc = id.bits; |
| uae_u16 msk, vmsk; |
| int pos = 0; |
| int mnp = 0; |
| int bitno = 0; |
| char mnemonic[64]; |
| |
| wordsizes sz = sz_long; |
| int srcgather = 0, dstgather = 0; |
| int usesrc = 0, usedst = 0; |
| int srctype = 0; |
| int srcpos = -1, dstpos = -1; |
| |
| amodes srcmode = am_unknown, destmode = am_unknown; |
| int srcreg = -1, destreg = -1; |
| |
| for (i = 0; i < lastbit; i++) |
| bitcnt[i] = bitval[i] = 0; |
| |
| vmsk = 1 << id.n_variable; |
| |
| for (i = 0, msk = 0x8000; i < 16; i++, msk >>= 1) { |
| if (!(msk & id.mask)) { |
| int currbit = id.bitpos[bitno++]; |
| int bit_set; |
| vmsk >>= 1; |
| bit_set = variants & vmsk ? 1 : 0; |
| if (bit_set) |
| opc |= msk; |
| bitpos[currbit] = 15 - i; |
| bitcnt[currbit]++; |
| bitval[currbit] <<= 1; |
| bitval[currbit] |= bit_set; |
| } |
| } |
| |
| if (bitval[bitj] == 0) bitval[bitj] = 8; |
| /* first check whether this one does not match after all */ |
| if (bitval[bitz] == 3 || bitval[bitC] == 1) |
| continue; |
| if (bitcnt[bitI] && (bitval[bitI] == 0x00 || bitval[bitI] == 0xff)) |
| continue; |
| if (bitcnt[bitE] && (bitval[bitE] == 0x00)) |
| continue; |
| |
| /* bitI and bitC get copied to biti and bitc */ |
| if (bitcnt[bitI]) { |
| bitval[biti] = bitval[bitI]; bitpos[biti] = bitpos[bitI]; |
| } |
| if (bitcnt[bitC]) |
| bitval[bitc] = bitval[bitC]; |
| |
| pos = 0; |
| while (opcstr[pos] && !isspace(opcstr[pos])) { |
| if (opcstr[pos] == '.') { |
| pos++; |
| switch (opcstr[pos]) { |
| |
| case 'B': sz = sz_byte; break; |
| case 'W': sz = sz_word; break; |
| case 'L': sz = sz_long; break; |
| case 'z': |
| switch (bitval[bitz]) { |
| case 0: sz = sz_byte; break; |
| case 1: sz = sz_word; break; |
| case 2: sz = sz_long; break; |
| default: abort(); |
| } |
| break; |
| default: abort(); |
| } |
| } else { |
| mnemonic[mnp] = opcstr[pos]; |
| if (mnemonic[mnp] == 'f') { |
| find = -1; |
| switch (bitval[bitf]) { |
| case 0: mnemonic[mnp] = 'R'; break; |
| case 1: mnemonic[mnp] = 'L'; break; |
| default: abort(); |
| } |
| } |
| mnp++; |
| if ((unsigned)mnp >= sizeof(mnemonic) - 1) { |
| mnemonic[sizeof(mnemonic) - 1] = 0; |
| fprintf(stderr, "Instruction %s overflow\n", mnemonic); |
| abort(); |
| } |
| } |
| pos++; |
| } |
| mnemonic[mnp] = 0; |
| |
| /* now, we have read the mnemonic and the size */ |
| while (opcstr[pos] && isspace(opcstr[pos])) |
| pos++; |
| |
| /* A goto a day keeps the D******a away. */ |
| if (opcstr[pos] == 0) |
| goto endofline; |
| |
| /* parse the source address */ |
| usesrc = 1; |
| switch (opcstr[pos++]) { |
| case 'D': |
| srcmode = Dreg; |
| switch (opcstr[pos++]) { |
| case 'r': srcreg = bitval[bitr]; srcgather = 1; srcpos = bitpos[bitr]; break; |
| case 'R': srcreg = bitval[bitR]; srcgather = 1; srcpos = bitpos[bitR]; break; |
| default: abort(); |
| } |
| |
| break; |
| case 'A': |
| srcmode = Areg; |
| switch (opcstr[pos++]) { |
| case 'r': srcreg = bitval[bitr]; srcgather = 1; srcpos = bitpos[bitr]; break; |
| case 'R': srcreg = bitval[bitR]; srcgather = 1; srcpos = bitpos[bitR]; break; |
| default: abort(); |
| } |
| switch (opcstr[pos]) { |
| case 'p': srcmode = Apdi; pos++; break; |
| case 'P': srcmode = Aipi; pos++; break; |
| } |
| break; |
| case 'L': |
| srcmode = absl; |
| break; |
| case '#': |
| switch (opcstr[pos++]) { |
| case 'z': srcmode = imm; break; |
| case '0': srcmode = imm0; break; |
| case '1': srcmode = imm1; break; |
| case '2': srcmode = imm2; break; |
| case 'i': srcmode = immi; srcreg = (uae_s32)(uae_s8)bitval[biti]; |
| if (CPU_EMU_SIZE < 4) { |
| /* Used for branch instructions */ |
| srctype = 1; |
| srcgather = 1; |
| srcpos = bitpos[biti]; |
| } |
| break; |
| case 'j': srcmode = immi; srcreg = bitval[bitj]; |
| if (CPU_EMU_SIZE < 3) { |
| /* 1..8 for ADDQ/SUBQ and rotshi insns */ |
| srcgather = 1; |
| srctype = 3; |
| srcpos = bitpos[bitj]; |
| } |
| break; |
| case 'J': srcmode = immi; srcreg = bitval[bitJ]; |
| if (CPU_EMU_SIZE < 5) { |
| /* 0..15 */ |
| srcgather = 1; |
| srctype = 2; |
| srcpos = bitpos[bitJ]; |
| } |
| break; |
| case 'k': srcmode = immi; srcreg = bitval[bitk]; |
| if (CPU_EMU_SIZE < 3) { |
| srcgather = 1; |
| srctype = 4; |
| srcpos = bitpos[bitk]; |
| } |
| break; |
| case 'K': srcmode = immi; srcreg = bitval[bitK]; |
| if (CPU_EMU_SIZE < 5) { |
| /* 0..15 */ |
| srcgather = 1; |
| srctype = 5; |
| srcpos = bitpos[bitK]; |
| } |
| break; |
| case 'E': srcmode = immi; srcreg = bitval[bitE]; |
| if (CPU_EMU_SIZE < 5) { // gb-- what is CPU_EMU_SIZE used for ?? |
| /* 1..255 */ |
| srcgather = 1; |
| srctype = 6; |
| srcpos = bitpos[bitE]; |
| } |
| break; |
| case 'p': srcmode = immi; srcreg = bitval[bitp]; |
| if (CPU_EMU_SIZE < 5) { |
| /* 0..3 */ |
| srcgather = 1; |
| srctype = 7; |
| srcpos = bitpos[bitp]; |
| } |
| break; |
| default: abort(); |
| } |
| break; |
| case 'd': |
| srcreg = bitval[bitD]; |
| srcmode = mode_from_mr(bitval[bitd],bitval[bitD]); |
| if (srcmode == am_illg) |
| continue; |
| if (CPU_EMU_SIZE < 2 && |
| (srcmode == Areg || srcmode == Dreg || srcmode == Aind |
| || srcmode == Ad16 || srcmode == Ad8r || srcmode == Aipi |
| || srcmode == Apdi)) |
| { |
| srcgather = 1; srcpos = bitpos[bitD]; |
| } |
| if (opcstr[pos] == '[') { |
| pos++; |
| if (opcstr[pos] == '!') { |
| /* exclusion */ |
| do { |
| pos++; |
| if (mode_from_str(opcstr+pos) == srcmode) |
| goto nomatch; |
| pos += 4; |
| } while (opcstr[pos] == ','); |
| pos++; |
| } else { |
| if (opcstr[pos+4] == '-') { |
| /* replacement */ |
| if (mode_from_str(opcstr+pos) == srcmode) |
| srcmode = mode_from_str(opcstr+pos+5); |
| else |
| goto nomatch; |
| pos += 10; |
| } else { |
| /* normal */ |
| while(mode_from_str(opcstr+pos) != srcmode) { |
| pos += 4; |
| if (opcstr[pos] == ']') |
| goto nomatch; |
| pos++; |
| } |
| while(opcstr[pos] != ']') pos++; |
| pos++; |
| break; |
| } |
| } |
| } |
| /* Some addressing modes are invalid as destination */ |
| if (srcmode == imm || srcmode == PC16 || srcmode == PC8r) |
| goto nomatch; |
| break; |
| case 's': |
| srcreg = bitval[bitS]; |
| srcmode = mode_from_mr(bitval[bits],bitval[bitS]); |
| |
| if (srcmode == am_illg) |
| continue; |
| if (CPU_EMU_SIZE < 2 && |
| (srcmode == Areg || srcmode == Dreg || srcmode == Aind |
| || srcmode == Ad16 || srcmode == Ad8r || srcmode == Aipi |
| || srcmode == Apdi)) |
| { |
| srcgather = 1; srcpos = bitpos[bitS]; |
| } |
| if (opcstr[pos] == '[') { |
| pos++; |
| if (opcstr[pos] == '!') { |
| /* exclusion */ |
| do { |
| pos++; |
| if (mode_from_str(opcstr+pos) == srcmode) |
| goto nomatch; |
| pos += 4; |
| } while (opcstr[pos] == ','); |
| pos++; |
| } else { |
| if (opcstr[pos+4] == '-') { |
| /* replacement */ |
| if (mode_from_str(opcstr+pos) == srcmode) |
| srcmode = mode_from_str(opcstr+pos+5); |
| else |
| goto nomatch; |
| pos += 10; |
| } else { |
| /* normal */ |
| while(mode_from_str(opcstr+pos) != srcmode) { |
| pos += 4; |
| if (opcstr[pos] == ']') |
| goto nomatch; |
| pos++; |
| } |
| while(opcstr[pos] != ']') pos++; |
| pos++; |
| } |
| } |
| } |
| break; |
| default: abort(); |
| } |
| /* safety check - might have changed */ |
| if (srcmode != Areg && srcmode != Dreg && srcmode != Aind |
| && srcmode != Ad16 && srcmode != Ad8r && srcmode != Aipi |
| && srcmode != Apdi && srcmode != immi) |
| { |
| srcgather = 0; |
| } |
| if (srcmode == Areg && sz == sz_byte) |
| goto nomatch; |
| |
| if (opcstr[pos] != ',') |
| goto endofline; |
| pos++; |
| |
| /* parse the destination address */ |
| usedst = 1; |
| switch (opcstr[pos++]) { |
| case 'D': |
| destmode = Dreg; |
| switch (opcstr[pos++]) { |
| case 'r': destreg = bitval[bitr]; dstgather = 1; dstpos = bitpos[bitr]; break; |
| case 'R': destreg = bitval[bitR]; dstgather = 1; dstpos = bitpos[bitR]; break; |
| default: abort(); |
| } |
| if (dstpos < 0 || dstpos >= 32) |
| abort(); |
| break; |
| case 'A': |
| destmode = Areg; |
| switch (opcstr[pos++]) { |
| case 'r': destreg = bitval[bitr]; dstgather = 1; dstpos = bitpos[bitr]; break; |
| case 'R': destreg = bitval[bitR]; dstgather = 1; dstpos = bitpos[bitR]; break; |
| case 'x': destreg = 0; dstgather = 0; dstpos = 0; break; |
| default: abort(); |
| } |
| if (dstpos < 0 || dstpos >= 32) |
| abort(); |
| switch (opcstr[pos]) { |
| case 'p': destmode = Apdi; pos++; break; |
| case 'P': destmode = Aipi; pos++; break; |
| } |
| break; |
| case 'L': |
| destmode = absl; |
| break; |
| case '#': |
| switch (opcstr[pos++]) { |
| case 'z': destmode = imm; break; |
| case '0': destmode = imm0; break; |
| case '1': destmode = imm1; break; |
| case '2': destmode = imm2; break; |
| case 'i': destmode = immi; destreg = (uae_s32)(uae_s8)bitval[biti]; break; |
| case 'j': destmode = immi; destreg = bitval[bitj]; break; |
| case 'J': destmode = immi; destreg = bitval[bitJ]; break; |
| case 'k': destmode = immi; destreg = bitval[bitk]; break; |
| case 'K': destmode = immi; destreg = bitval[bitK]; break; |
| default: abort(); |
| } |
| break; |
| case 'd': |
| destreg = bitval[bitD]; |
| destmode = mode_from_mr(bitval[bitd],bitval[bitD]); |
| if (destmode == am_illg) |
| continue; |
| if (CPU_EMU_SIZE < 1 && |
| (destmode == Areg || destmode == Dreg || destmode == Aind |
| || destmode == Ad16 || destmode == Ad8r || destmode == Aipi |
| || destmode == Apdi)) |
| { |
| dstgather = 1; dstpos = bitpos[bitD]; |
| } |
| |
| if (opcstr[pos] == '[') { |
| pos++; |
| if (opcstr[pos] == '!') { |
| /* exclusion */ |
| do { |
| pos++; |
| if (mode_from_str(opcstr+pos) == destmode) |
| goto nomatch; |
| pos += 4; |
| } while (opcstr[pos] == ','); |
| pos++; |
| } else { |
| if (opcstr[pos+4] == '-') { |
| /* replacement */ |
| if (mode_from_str(opcstr+pos) == destmode) |
| destmode = mode_from_str(opcstr+pos+5); |
| else |
| goto nomatch; |
| pos += 10; |
| } else { |
| /* normal */ |
| while(mode_from_str(opcstr+pos) != destmode) { |
| pos += 4; |
| if (opcstr[pos] == ']') |
| goto nomatch; |
| pos++; |
| } |
| while(opcstr[pos] != ']') pos++; |
| pos++; |
| break; |
| } |
| } |
| } |
| /* Some addressing modes are invalid as destination */ |
| if (destmode == imm || destmode == PC16 || destmode == PC8r) |
| goto nomatch; |
| break; |
| case 's': |
| destreg = bitval[bitS]; |
| destmode = mode_from_mr(bitval[bits],bitval[bitS]); |
| |
| if (destmode == am_illg) |
| continue; |
| if (CPU_EMU_SIZE < 1 && |
| (destmode == Areg || destmode == Dreg || destmode == Aind |
| || destmode == Ad16 || destmode == Ad8r || destmode == Aipi |
| || destmode == Apdi)) |
| { |
| dstgather = 1; dstpos = bitpos[bitS]; |
| } |
| |
| if (opcstr[pos] == '[') { |
| pos++; |
| if (opcstr[pos] == '!') { |
| /* exclusion */ |
| do { |
| pos++; |
| if (mode_from_str(opcstr+pos) == destmode) |
| goto nomatch; |
| pos += 4; |
| } while (opcstr[pos] == ','); |
| pos++; |
| } else { |
| if (opcstr[pos+4] == '-') { |
| /* replacement */ |
| if (mode_from_str(opcstr+pos) == destmode) |
| destmode = mode_from_str(opcstr+pos+5); |
| else |
| goto nomatch; |
| pos += 10; |
| } else { |
| /* normal */ |
| while(mode_from_str(opcstr+pos) != destmode) { |
| pos += 4; |
| if (opcstr[pos] == ']') |
| goto nomatch; |
| pos++; |
| } |
| while(opcstr[pos] != ']') pos++; |
| pos++; |
| } |
| } |
| } |
| break; |
| default: abort(); |
| } |
| /* safety check - might have changed */ |
| if (destmode != Areg && destmode != Dreg && destmode != Aind |
| && destmode != Ad16 && destmode != Ad8r && destmode != Aipi |
| && destmode != Apdi) |
| { |
| dstgather = 0; |
| } |
| |
| if (destmode == Areg && sz == sz_byte) |
| goto nomatch; |
| #if 0 |
| if (sz == sz_byte && (destmode == Aipi || destmode == Apdi)) { |
| dstgather = 0; |
| } |
| #endif |
| endofline: |
| /* now, we have a match */ |
| if (table68k[opc].mnemo != i_ILLG) |
| fprintf(stderr, "Double match: %x: %s\n", opc, opcstr); |
| if (find == -1) { |
| for (find = 0;; find++) { |
| if (strcmp(mnemonic, lookuptab[find].name) == 0) { |
| table68k[opc].mnemo = lookuptab[find].mnemo; |
| break; |
| } |
| if (strlen(lookuptab[find].name) == 0) abort(); |
| } |
| } |
| else { |
| table68k[opc].mnemo = lookuptab[find].mnemo; |
| } |
| table68k[opc].cc = bitval[bitc]; |
| if (table68k[opc].mnemo == i_BTST |
| || table68k[opc].mnemo == i_BSET |
| || table68k[opc].mnemo == i_BCLR |
| || table68k[opc].mnemo == i_BCHG) |
| { |
| sz = destmode == Dreg ? sz_long : sz_byte; |
| } |
| table68k[opc].size = sz; |
| table68k[opc].sreg = srcreg; |
| table68k[opc].dreg = destreg; |
| table68k[opc].smode = srcmode; |
| table68k[opc].dmode = destmode; |
| table68k[opc].spos = srcgather ? srcpos : -1; |
| table68k[opc].dpos = dstgather ? dstpos : -1; |
| table68k[opc].suse = usesrc; |
| table68k[opc].duse = usedst; |
| table68k[opc].stype = srctype; |
| table68k[opc].plev = id.plevel; |
| table68k[opc].clev = id.cpulevel; |
| #if 0 |
| for (i = 0; i < 5; i++) { |
| table68k[opc].flaginfo[i].flagset = id.flaginfo[i].flagset; |
| table68k[opc].flaginfo[i].flaguse = id.flaginfo[i].flaguse; |
| } |
| #endif |
| |
| // Fix flags used information for Scc, Bcc, TRAPcc, DBcc instructions |
| if ( table68k[opc].mnemo == i_Scc |
| || table68k[opc].mnemo == i_Bcc |
| || table68k[opc].mnemo == i_DBcc |
| || table68k[opc].mnemo == i_TRAPcc |
| ) { |
| switch (table68k[opc].cc) { |
| // CC mask: XNZVC |
| // 8421 |
| case 0: flags_used = 0x00; break; /* T */ |
| case 1: flags_used = 0x00; break; /* F */ |
| case 2: flags_used = 0x05; break; /* HI */ |
| case 3: flags_used = 0x05; break; /* LS */ |
| case 4: flags_used = 0x01; break; /* CC */ |
| case 5: flags_used = 0x01; break; /* CS */ |
| case 6: flags_used = 0x04; break; /* NE */ |
| case 7: flags_used = 0x04; break; /* EQ */ |
| case 8: flags_used = 0x02; break; /* VC */ |
| case 9: flags_used = 0x02; break; /* VS */ |
| case 10:flags_used = 0x08; break; /* PL */ |
| case 11:flags_used = 0x08; break; /* MI */ |
| case 12:flags_used = 0x0A; break; /* GE */ |
| case 13:flags_used = 0x0A; break; /* LT */ |
| case 14:flags_used = 0x0E; break; /* GT */ |
| case 15:flags_used = 0x0E; break; /* LE */ |
| } |
| } |
| |
| #if 1 |
| /* gb-- flagdead and flaglive would not have correct information */ |
| table68k[opc].flagdead = flags_set; |
| table68k[opc].flaglive = flags_used; |
| #else |
| table68k[opc].flagdead = flagdead; |
| table68k[opc].flaglive = flaglive; |
| #endif |
| table68k[opc].cflow = cflow; |
| nomatch: |
| /* FOO! */; |
| } |
| } |
| |
| |
| void read_table68k (void) |
| { |
| int i; |
| |
| table68k = (struct instr *)malloc (65536 * sizeof (struct instr)); |
| for (i = 0; i < 65536; i++) { |
| table68k[i].mnemo = i_ILLG; |
| table68k[i].handler = -1; |
| } |
| for (i = 0; i < n_defs68k; i++) { |
| build_insn (i); |
| } |
| } |
| |
| static int mismatch; |
| |
| static void handle_merges (long int opcode) |
| { |
| uae_u16 smsk; |
| uae_u16 dmsk; |
| int sbitdst, dstend; |
| int srcreg, dstreg; |
| |
| if (table68k[opcode].spos == -1) { |
| sbitdst = 1; smsk = 0; |
| } else { |
| switch (table68k[opcode].stype) { |
| case 0: |
| smsk = 7; sbitdst = 8; break; |
| case 1: |
| smsk = 255; sbitdst = 256; break; |
| case 2: |
| smsk = 15; sbitdst = 16; break; |
| case 3: |
| smsk = 7; sbitdst = 8; break; |
| case 4: |
| smsk = 7; sbitdst = 8; break; |
| case 5: |
| smsk = 63; sbitdst = 64; break; |
| case 6: |
| smsk = 255; sbitdst = 256; break; |
| case 7: |
| smsk = 3; sbitdst = 4; break; |
| default: |
| smsk = 0; sbitdst = 0; |
| abort(); |
| break; |
| } |
| smsk <<= table68k[opcode].spos; |
| } |
| if (table68k[opcode].dpos == -1) { |
| dstend = 1; dmsk = 0; |
| } else { |
| dmsk = 7 << table68k[opcode].dpos; |
| dstend = 8; |
| } |
| for (srcreg=0; srcreg < sbitdst; srcreg++) { |
| for (dstreg=0; dstreg < dstend; dstreg++) { |
| uae_u16 code = uae_u16(opcode); |
| |
| code = (code & ~smsk) | (srcreg << table68k[opcode].spos); |
| code = (code & ~dmsk) | (dstreg << table68k[opcode].dpos); |
| |
| /* Check whether this is in fact the same instruction. |
| * The instructions should never differ, except for the |
| * Bcc.(BW) case. */ |
| if (table68k[code].mnemo != table68k[opcode].mnemo |
| || table68k[code].size != table68k[opcode].size |
| || table68k[code].suse != table68k[opcode].suse |
| || table68k[code].duse != table68k[opcode].duse) |
| { |
| mismatch++; continue; |
| } |
| if (table68k[opcode].suse |
| && (table68k[opcode].spos != table68k[code].spos |
| || table68k[opcode].smode != table68k[code].smode |
| || table68k[opcode].stype != table68k[code].stype)) |
| { |
| mismatch++; continue; |
| } |
| if (table68k[opcode].duse |
| && (table68k[opcode].dpos != table68k[code].dpos |
| || table68k[opcode].dmode != table68k[code].dmode)) |
| { |
| mismatch++; continue; |
| } |
| |
| if (code != opcode) |
| table68k[code].handler = opcode; |
| } |
| } |
| } |
| |
| void do_merges (void) |
| { |
| long int opcode; |
| int nr = 0; |
| mismatch = 0; |
| for (opcode = 0; opcode < 65536; opcode++) { |
| if (table68k[opcode].handler != -1 || table68k[opcode].mnemo == i_ILLG) |
| continue; |
| nr++; |
| handle_merges (opcode); |
| } |
| nr_cpuop_funcs = nr; |
| } |
| |
| int get_no_mismatches (void) |
| { |
| return mismatch; |
| } |
| |
| const char *get_instruction_name (unsigned int opcode) |
| { |
| struct instr *ins = &table68k[opcode]; |
| for (int i = 0; lookuptab[i].name[0]; i++) { |
| if (ins->mnemo == lookuptab[i].mnemo) |
| return lookuptab[i].name; |
| } |
| abort(); |
| return NULL; |
| } |
| |
| static char *get_ea_string (amodes mode, wordsizes size) |
| { |
| static char buffer[80]; |
| |
| buffer[0] = 0; |
| switch (mode){ |
| case Dreg: |
| strcpy (buffer,"Dn"); |
| break; |
| case Areg: |
| strcpy (buffer,"An"); |
| break; |
| case Aind: |
| strcpy (buffer,"(An)"); |
| break; |
| case Aipi: |
| strcpy (buffer,"(An)+"); |
| break; |
| case Apdi: |
| strcpy (buffer,"-(An)"); |
| break; |
| case Ad16: |
| strcpy (buffer,"(d16,An)"); |
| break; |
| case Ad8r: |
| strcpy (buffer,"(d8,An,Xn)"); |
| break; |
| case PC16: |
| strcpy (buffer,"(d16,PC)"); |
| break; |
| case PC8r: |
| strcpy (buffer,"(d8,PC,Xn)"); |
| break; |
| case absw: |
| strcpy (buffer,"(xxx).W"); |
| break; |
| case absl: |
| strcpy (buffer,"(xxx).L"); |
| break; |
| case imm: |
| switch (size){ |
| case sz_byte: |
| strcpy (buffer,"#<data>.B"); |
| break; |
| case sz_word: |
| strcpy (buffer,"#<data>.W"); |
| break; |
| case sz_long: |
| strcpy (buffer,"#<data>.L"); |
| break; |
| default: |
| break; |
| } |
| break; |
| case imm0: |
| strcpy (buffer,"#<data>.B"); |
| break; |
| case imm1: |
| strcpy (buffer,"#<data>.W"); |
| break; |
| case imm2: |
| strcpy (buffer,"#<data>.L"); |
| break; |
| case immi: |
| strcpy (buffer,"#<data>"); |
| break; |
| |
| default: |
| break; |
| } |
| return buffer; |
| } |
| |
| const char *get_instruction_string (unsigned int opcode) |
| { |
| static char out[100]; |
| struct instr *ins; |
| |
| strcpy (out, get_instruction_name (opcode)); |
| |
| ins = &table68k[opcode]; |
| if (ins->size == sz_byte) |
| strcat (out,".B"); |
| if (ins->size == sz_word) |
| strcat (out,".W"); |
| if (ins->size == sz_long) |
| strcat (out,".L"); |
| strcat (out," "); |
| if (ins->suse) |
| strcat (out, get_ea_string (amodes(ins->smode), wordsizes(ins->size))); |
| if (ins->duse) { |
| if (ins->suse) |
| strcat (out,","); |
| strcat (out, get_ea_string (amodes(ins->dmode), wordsizes(ins->size))); |
| } |
| return out; |
| } |