File: disassembler.cpp

package info (click to toggle)
libretro-bsnes-mercury 094%2Bgit20220807-8
  • links: PTS, VCS
  • area: main
  • in suites: bookworm, forky, sid, trixie
  • size: 9,792 kB
  • sloc: cpp: 109,408; ansic: 3,554; makefile: 991; xml: 115; asm: 55; sh: 1
file content (196 lines) | stat: -rw-r--r-- 6,720 bytes parent folder | download | duplicates (5)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
#ifdef NECDSP_CPP

string NECDSP::disassemble(uint14 ip) {
  string output = { hex<4>(ip), "  " };
  uint24 opcode = programROM[ip];
  uint2 type = opcode >> 22;

  if(type == 0 || type == 1) {  //OP,RT
    uint2 pselect = opcode >> 20;
    uint4 alu     = opcode >> 16;
    uint1 asl     = opcode >> 15;
    uint2 dpl     = opcode >> 13;
    uint4 dphm    = opcode >>  9;
    uint1 rpdcr   = opcode >>  8;
    uint4 src     = opcode >>  4;
    uint4 dst     = opcode >>  0;

    switch(alu) {
    case  0: output.append("nop     "); break;
    case  1: output.append("or      "); break;
    case  2: output.append("and     "); break;
    case  3: output.append("xor     "); break;
    case  4: output.append("sub     "); break;
    case  5: output.append("add     "); break;
    case  6: output.append("sbb     "); break;
    case  7: output.append("adc     "); break;
    case  8: output.append("dec     "); break;
    case  9: output.append("inc     "); break;
    case 10: output.append("cmp     "); break;
    case 11: output.append("shr1    "); break;
    case 12: output.append("shl1    "); break;
    case 13: output.append("shl2    "); break;
    case 14: output.append("shl4    "); break;
    case 15: output.append("xchg    "); break;
    }

    if(alu < 8) {
      switch(pselect) {
      case 0: output.append("ram,"); break;
      case 1: output.append("idb,"); break;
      case 2: output.append("m,"  ); break;
      case 3: output.append("n,"  ); break;
      }
    }

    switch(asl) {
    case 0: output.append("a"); break;
    case 1: output.append("b"); break;
    }

    output.append("\n      mov     ");

    switch(src) {
    case  0: output.append("trb," ); break;
    case  1: output.append("a,"   ); break;
    case  2: output.append("b,"   ); break;
    case  3: output.append("tr,"  ); break;
    case  4: output.append("dp,"  ); break;
    case  5: output.append("rp,"  ); break;
    case  6: output.append("ro,"  ); break;
    case  7: output.append("sgn," ); break;
    case  8: output.append("dr,"  ); break;
    case  9: output.append("drnf,"); break;
    case 10: output.append("sr,"  ); break;
    case 11: output.append("sim," ); break;
    case 12: output.append("sil," ); break;
    case 13: output.append("k,"   ); break;
    case 14: output.append("l,"   ); break;
    case 15: output.append("mem," ); break;
    }

    switch(dst) {
    case  0: output.append("non"); break;
    case  1: output.append("a"  ); break;
    case  2: output.append("b"  ); break;
    case  3: output.append("tr" ); break;
    case  4: output.append("dp" ); break;
    case  5: output.append("rp" ); break;
    case  6: output.append("dr" ); break;
    case  7: output.append("sr" ); break;
    case  8: output.append("sol"); break;
    case  9: output.append("som"); break;
    case 10: output.append("k"  ); break;
    case 11: output.append("klr"); break;
    case 12: output.append("klm"); break;
    case 13: output.append("l"  ); break;
    case 14: output.append("trb"); break;
    case 15: output.append("mem"); break;
    }

    if(dpl) {
      switch(dpl) {
      case 0: output.append("\n      dpnop"); break;
      case 1: output.append("\n      dpinc"); break;
      case 2: output.append("\n      dpdec"); break;
      case 3: output.append("\n      dpclr"); break;
      }
    }

    if(dphm) {
      output.append("\n      m", hex<1>(dphm));
    }

    if(rpdcr == 1) {
      output.append("\n      rpdec");
    }

    if(type == 1) {
      output.append("\n      ret");
    }
  }

  if(type == 2) {  //JP
    uint9 brch = opcode >> 13;
    uint11 na  = opcode >>  2;
    uint8 bank = opcode >>  0;

    uint14 jp = (regs.pc & 0x2000) | (bank << 11) | (na << 0);

    switch(brch) {
    case 0x000: output.append("jmpso   "); jp = 0; break;
    case 0x080: output.append("jnca    "); break;
    case 0x082: output.append("jca     "); break;
    case 0x084: output.append("jncb    "); break;
    case 0x086: output.append("jcb     "); break;
    case 0x088: output.append("jnza    "); break;
    case 0x08a: output.append("jza     "); break;
    case 0x08c: output.append("jnzb    "); break;
    case 0x08e: output.append("jzb     "); break;
    case 0x090: output.append("jnova0  "); break;
    case 0x092: output.append("jova0   "); break;
    case 0x094: output.append("jnovb0  "); break;
    case 0x096: output.append("jovb0   "); break;
    case 0x098: output.append("jnova1  "); break;
    case 0x09a: output.append("jova1   "); break;
    case 0x09c: output.append("jnovb1  "); break;
    case 0x09e: output.append("jovb1   "); break;
    case 0x0a0: output.append("jnsa0   "); break;
    case 0x0a2: output.append("jsa0    "); break;
    case 0x0a4: output.append("jnsb0   "); break;
    case 0x0a6: output.append("jsb0    "); break;
    case 0x0a8: output.append("jnsa1   "); break;
    case 0x0aa: output.append("jsa1    "); break;
    case 0x0ac: output.append("jnsb1   "); break;
    case 0x0ae: output.append("jsb1    "); break;
    case 0x0b0: output.append("jdpl0   "); break;
    case 0x0b1: output.append("jdpln0  "); break;
    case 0x0b2: output.append("jdplf   "); break;
    case 0x0b3: output.append("jdplnf  "); break;
    case 0x0b4: output.append("jnsiak  "); break;
    case 0x0b6: output.append("jsiak   "); break;
    case 0x0b8: output.append("jnsoak  "); break;
    case 0x0ba: output.append("jsoak   "); break;
    case 0x0bc: output.append("jnrqm   "); break;
    case 0x0be: output.append("jrqm    "); break;
    case 0x100: output.append("ljmp    "); jp &= ~0x2000; break;
    case 0x101: output.append("hjmp    "); jp |=  0x2000; break;
    case 0x140: output.append("lcall   "); jp &= ~0x2000; break;
    case 0x141: output.append("hcall   "); jp |=  0x2000; break;
    default:    output.append("??????  "); break;
    }

    output.append("$", hex<4>(jp));
  }

  if(type == 3) {  //LD
    output.append("ld      ");
    uint16 id = opcode >> 6;
    uint4 dst = opcode >> 0;

    output.append("$", hex<4>(id), ",");

    switch(dst) {
    case  0: output.append("non"); break;
    case  1: output.append("a"  ); break;
    case  2: output.append("b"  ); break;
    case  3: output.append("tr" ); break;
    case  4: output.append("dp" ); break;
    case  5: output.append("rp" ); break;
    case  6: output.append("dr" ); break;
    case  7: output.append("sr" ); break;
    case  8: output.append("sol"); break;
    case  9: output.append("som"); break;
    case 10: output.append("k"  ); break;
    case 11: output.append("klr"); break;
    case 12: output.append("klm"); break;
    case 13: output.append("l"  ); break;
    case 14: output.append("trb"); break;
    case 15: output.append("mem"); break;
    }
  }

  return output;
}

#endif