1 |
cebix |
1.1 |
/* |
2 |
cebix |
1.2 |
* CPU_emulline.h - 6510/6502 emulation core (body of |
3 |
cebix |
1.1 |
* EmulateLine() function, the same for |
4 |
|
|
* both 6510 and 6502) |
5 |
|
|
* |
6 |
|
|
* Frodo (C) 1994-1997,2002 Christian Bauer |
7 |
|
|
* |
8 |
|
|
* This program is free software; you can redistribute it and/or modify |
9 |
|
|
* it under the terms of the GNU General Public License as published by |
10 |
|
|
* the Free Software Foundation; either version 2 of the License, or |
11 |
|
|
* (at your option) any later version. |
12 |
|
|
* |
13 |
|
|
* This program is distributed in the hope that it will be useful, |
14 |
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of |
15 |
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the |
16 |
|
|
* GNU General Public License for more details. |
17 |
|
|
* |
18 |
|
|
* You should have received a copy of the GNU General Public License |
19 |
|
|
* along with this program; if not, write to the Free Software |
20 |
|
|
* Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA |
21 |
|
|
*/ |
22 |
|
|
|
23 |
|
|
|
24 |
|
|
/* |
25 |
|
|
* Addressing mode macros |
26 |
|
|
*/ |
27 |
|
|
|
28 |
|
|
// Read immediate operand |
29 |
|
|
#if PC_IS_POINTER |
30 |
|
|
#define read_byte_imm() (*pc++) |
31 |
|
|
#else |
32 |
|
|
#define read_byte_imm() read_byte(pc++) |
33 |
|
|
#endif |
34 |
|
|
|
35 |
|
|
// Read zeropage operand address |
36 |
|
|
#define read_adr_zero() ((uint16)read_byte_imm()) |
37 |
|
|
|
38 |
|
|
// Read zeropage x-indexed operand address |
39 |
|
|
#define read_adr_zero_x() ((read_byte_imm() + x) & 0xff) |
40 |
|
|
|
41 |
|
|
// Read zeropage y-indexed operand address |
42 |
|
|
#define read_adr_zero_y() ((read_byte_imm() + y) & 0xff) |
43 |
|
|
|
44 |
|
|
// Read absolute operand address (uses adr!) |
45 |
|
|
#if PC_IS_POINTER |
46 |
|
|
#if LITTLE_ENDIAN_UNALIGNED |
47 |
|
|
#define read_adr_abs() (adr = *(UWORD *)pc, pc+=2, adr) |
48 |
|
|
#else |
49 |
|
|
#define read_adr_abs() (adr = ((*(pc+1)) << 8) | *pc, pc+=2, adr) |
50 |
|
|
#endif |
51 |
|
|
#else |
52 |
|
|
#define read_adr_abs() (adr = read_word(pc), pc+=2, adr) |
53 |
|
|
#endif |
54 |
|
|
|
55 |
|
|
// Read absolute x-indexed operand address |
56 |
|
|
#define read_adr_abs_x() (read_adr_abs() + x) |
57 |
|
|
|
58 |
|
|
// Read absolute y-indexed operand address |
59 |
|
|
#define read_adr_abs_y() (read_adr_abs() + y) |
60 |
|
|
|
61 |
|
|
// Read indexed indirect operand address |
62 |
|
|
#define read_adr_ind_x() (read_zp_word(read_byte_imm() + x)) |
63 |
|
|
|
64 |
|
|
// Read indirect indexed operand address |
65 |
|
|
#define read_adr_ind_y() (read_zp_word(read_byte_imm()) + y) |
66 |
|
|
|
67 |
|
|
// Read zeropage operand |
68 |
|
|
#define read_byte_zero() read_zp(read_adr_zero()) |
69 |
|
|
|
70 |
|
|
// Read zeropage x-indexed operand |
71 |
|
|
#define read_byte_zero_x() read_zp(read_adr_zero_x()) |
72 |
|
|
|
73 |
|
|
// Read zeropage y-indexed operand |
74 |
|
|
#define read_byte_zero_y() read_zp(read_adr_zero_y()) |
75 |
|
|
|
76 |
|
|
// Read absolute operand |
77 |
|
|
#define read_byte_abs() read_byte(read_adr_abs()) |
78 |
|
|
|
79 |
|
|
#if PRECISE_CPU_CYCLES |
80 |
|
|
// Acount for cyles due to crossing page boundaries |
81 |
|
|
#define page_plus(exp, reg) \ |
82 |
|
|
(adr = exp, page_cycles = (adr & 0xff) + reg >= 0x100, adr + reg) |
83 |
|
|
|
84 |
|
|
// Read absolute x-indexed operand |
85 |
|
|
#define read_byte_abs_x() read_byte(page_plus(read_adr_abs(), x)) |
86 |
|
|
|
87 |
|
|
// Read absolute x-indexed operand |
88 |
|
|
#define read_byte_abs_y() read_byte(page_plus(read_adr_abs(), y)) |
89 |
|
|
|
90 |
|
|
// Read indirect y-indexed operand |
91 |
|
|
#define read_byte_ind_y() read_byte(page_plus(read_zp_word(read_byte_imm()), y)) |
92 |
|
|
|
93 |
|
|
#else |
94 |
|
|
|
95 |
|
|
// Read absolute x-indexed operand |
96 |
|
|
#define read_byte_abs_x() read_byte(read_adr_abs_x()) |
97 |
|
|
|
98 |
|
|
// Read absolute x-indexed operand |
99 |
|
|
#define read_byte_abs_y() read_byte(read_adr_abs_y()) |
100 |
|
|
|
101 |
|
|
// Read indirect y-indexed operand |
102 |
|
|
#define read_byte_ind_y() read_byte(read_adr_ind_y()) |
103 |
|
|
#endif |
104 |
|
|
|
105 |
|
|
// Read indexed indirect operand |
106 |
|
|
#define read_byte_ind_x() read_byte(read_adr_ind_x()) |
107 |
|
|
|
108 |
|
|
|
109 |
|
|
/* |
110 |
|
|
* Set N and Z flags according to byte |
111 |
|
|
*/ |
112 |
|
|
|
113 |
|
|
#define set_nz(x) (z_flag = n_flag = (x)) |
114 |
|
|
|
115 |
|
|
|
116 |
|
|
/* |
117 |
|
|
* End of opcode, decrement cycles left |
118 |
|
|
*/ |
119 |
|
|
|
120 |
|
|
#define ENDOP(cyc) last_cycles = cyc; break; |
121 |
|
|
|
122 |
|
|
|
123 |
|
|
// Main opcode fetch/execute loop |
124 |
|
|
#if PRECISE_CPU_CYCLES |
125 |
|
|
if (cycles_left != 1) |
126 |
|
|
cycles_left -= borrowed_cycles; |
127 |
|
|
int page_cycles = 0; |
128 |
|
|
for (;;) { |
129 |
|
|
if (last_cycles) { |
130 |
|
|
last_cycles += page_cycles; |
131 |
|
|
page_cycles = 0; |
132 |
|
|
#if PRECISE_CIA_CYCLES && !defined(IS_CPU_1541) |
133 |
|
|
TheCIA1->EmulateLine(last_cycles); |
134 |
|
|
TheCIA2->EmulateLine(last_cycles); |
135 |
|
|
#endif |
136 |
|
|
} |
137 |
|
|
if ((cycles_left -= last_cycles) < 0) { |
138 |
|
|
borrowed_cycles = -cycles_left; |
139 |
|
|
break; |
140 |
|
|
} |
141 |
|
|
#else |
142 |
|
|
while ((cycles_left -= last_cycles) >= 0) { |
143 |
|
|
#endif |
144 |
|
|
|
145 |
|
|
switch (read_byte_imm()) { |
146 |
|
|
|
147 |
|
|
|
148 |
|
|
// Load group |
149 |
|
|
case 0xa9: // LDA #imm |
150 |
|
|
set_nz(a = read_byte_imm()); |
151 |
|
|
ENDOP(2); |
152 |
|
|
|
153 |
|
|
case 0xa5: // LDA zero |
154 |
|
|
set_nz(a = read_byte_zero()); |
155 |
|
|
ENDOP(3); |
156 |
|
|
|
157 |
|
|
case 0xb5: // LDA zero,X |
158 |
|
|
set_nz(a = read_byte_zero_x()); |
159 |
|
|
ENDOP(4); |
160 |
|
|
|
161 |
|
|
case 0xad: // LDA abs |
162 |
|
|
set_nz(a = read_byte_abs()); |
163 |
|
|
ENDOP(4); |
164 |
|
|
|
165 |
|
|
case 0xbd: // LDA abs,X |
166 |
|
|
set_nz(a = read_byte_abs_x()); |
167 |
|
|
ENDOP(4); |
168 |
|
|
|
169 |
|
|
case 0xb9: // LDA abs,Y |
170 |
|
|
set_nz(a = read_byte_abs_y()); |
171 |
|
|
ENDOP(4); |
172 |
|
|
|
173 |
|
|
case 0xa1: // LDA (ind,X) |
174 |
|
|
set_nz(a = read_byte_ind_x()); |
175 |
|
|
ENDOP(6); |
176 |
|
|
|
177 |
|
|
case 0xb1: // LDA (ind),Y |
178 |
|
|
set_nz(a = read_byte_ind_y()); |
179 |
|
|
ENDOP(5); |
180 |
|
|
|
181 |
|
|
case 0xa2: // LDX #imm |
182 |
|
|
set_nz(x = read_byte_imm()); |
183 |
|
|
ENDOP(2); |
184 |
|
|
|
185 |
|
|
case 0xa6: // LDX zero |
186 |
|
|
set_nz(x = read_byte_zero()); |
187 |
|
|
ENDOP(3); |
188 |
|
|
|
189 |
|
|
case 0xb6: // LDX zero,Y |
190 |
|
|
set_nz(x = read_byte_zero_y()); |
191 |
|
|
ENDOP(4); |
192 |
|
|
|
193 |
|
|
case 0xae: // LDX abs |
194 |
|
|
set_nz(x = read_byte_abs()); |
195 |
|
|
ENDOP(4); |
196 |
|
|
|
197 |
|
|
case 0xbe: // LDX abs,Y |
198 |
|
|
set_nz(x = read_byte_abs_y()); |
199 |
|
|
ENDOP(4); |
200 |
|
|
|
201 |
|
|
case 0xa0: // LDY #imm |
202 |
|
|
set_nz(y = read_byte_imm()); |
203 |
|
|
ENDOP(2); |
204 |
|
|
|
205 |
|
|
case 0xa4: // LDY zero |
206 |
|
|
set_nz(y = read_byte_zero()); |
207 |
|
|
ENDOP(3); |
208 |
|
|
|
209 |
|
|
case 0xb4: // LDY zero,X |
210 |
|
|
set_nz(y = read_byte_zero_x()); |
211 |
|
|
ENDOP(4); |
212 |
|
|
|
213 |
|
|
case 0xac: // LDY abs |
214 |
|
|
set_nz(y = read_byte_abs()); |
215 |
|
|
ENDOP(4); |
216 |
|
|
|
217 |
|
|
case 0xbc: // LDY abs,X |
218 |
|
|
set_nz(y = read_byte_abs_x()); |
219 |
|
|
ENDOP(4); |
220 |
|
|
|
221 |
|
|
|
222 |
|
|
// Store group |
223 |
|
|
case 0x85: // STA zero |
224 |
|
|
write_byte(read_adr_zero(), a); |
225 |
|
|
ENDOP(3); |
226 |
|
|
|
227 |
|
|
case 0x95: // STA zero,X |
228 |
|
|
write_byte(read_adr_zero_x(), a); |
229 |
|
|
ENDOP(4); |
230 |
|
|
|
231 |
|
|
case 0x8d: // STA abs |
232 |
|
|
write_byte(read_adr_abs(), a); |
233 |
|
|
ENDOP(4); |
234 |
|
|
|
235 |
|
|
case 0x9d: // STA abs,X |
236 |
|
|
write_byte(read_adr_abs_x(), a); |
237 |
|
|
ENDOP(5); |
238 |
|
|
|
239 |
|
|
case 0x99: // STA abs,Y |
240 |
|
|
write_byte(read_adr_abs_y(), a); |
241 |
|
|
ENDOP(5); |
242 |
|
|
|
243 |
|
|
case 0x81: // STA (ind,X) |
244 |
|
|
write_byte(read_adr_ind_x(), a); |
245 |
|
|
ENDOP(6); |
246 |
|
|
|
247 |
|
|
case 0x91: // STA (ind),Y |
248 |
|
|
write_byte(read_adr_ind_y(), a); |
249 |
|
|
ENDOP(6); |
250 |
|
|
|
251 |
|
|
case 0x86: // STX zero |
252 |
|
|
write_byte(read_adr_zero(), x); |
253 |
|
|
ENDOP(3); |
254 |
|
|
|
255 |
|
|
case 0x96: // STX zero,Y |
256 |
|
|
write_byte(read_adr_zero_y(), x); |
257 |
|
|
ENDOP(4); |
258 |
|
|
|
259 |
|
|
case 0x8e: // STX abs |
260 |
|
|
write_byte(read_adr_abs(), x); |
261 |
|
|
ENDOP(4); |
262 |
|
|
|
263 |
|
|
case 0x84: // STY zero |
264 |
|
|
write_byte(read_adr_zero(), y); |
265 |
|
|
ENDOP(3); |
266 |
|
|
|
267 |
|
|
case 0x94: // STY zero,X |
268 |
|
|
write_byte(read_adr_zero_x(), y); |
269 |
|
|
ENDOP(4); |
270 |
|
|
|
271 |
|
|
case 0x8c: // STY abs |
272 |
|
|
write_byte(read_adr_abs(), y); |
273 |
|
|
ENDOP(4); |
274 |
|
|
|
275 |
|
|
|
276 |
|
|
// Transfer group |
277 |
|
|
case 0xaa: // TAX |
278 |
|
|
set_nz(x = a); |
279 |
|
|
ENDOP(2); |
280 |
|
|
|
281 |
|
|
case 0x8a: // TXA |
282 |
|
|
set_nz(a = x); |
283 |
|
|
ENDOP(2); |
284 |
|
|
|
285 |
|
|
case 0xa8: // TAY |
286 |
|
|
set_nz(y = a); |
287 |
|
|
ENDOP(2); |
288 |
|
|
|
289 |
|
|
case 0x98: // TYA |
290 |
|
|
set_nz(a = y); |
291 |
|
|
ENDOP(2); |
292 |
|
|
|
293 |
|
|
case 0xba: // TSX |
294 |
|
|
set_nz(x = sp); |
295 |
|
|
ENDOP(2); |
296 |
|
|
|
297 |
|
|
case 0x9a: // TXS |
298 |
|
|
sp = x; |
299 |
|
|
ENDOP(2); |
300 |
|
|
|
301 |
|
|
|
302 |
|
|
// Arithmetic group |
303 |
|
|
case 0x69: // ADC #imm |
304 |
|
|
do_adc(read_byte_imm()); |
305 |
|
|
ENDOP(2); |
306 |
|
|
|
307 |
|
|
case 0x65: // ADC zero |
308 |
|
|
do_adc(read_byte_zero()); |
309 |
|
|
ENDOP(3); |
310 |
|
|
|
311 |
|
|
case 0x75: // ADC zero,X |
312 |
|
|
do_adc(read_byte_zero_x()); |
313 |
|
|
ENDOP(4); |
314 |
|
|
|
315 |
|
|
case 0x6d: // ADC abs |
316 |
|
|
do_adc(read_byte_abs()); |
317 |
|
|
ENDOP(4); |
318 |
|
|
|
319 |
|
|
case 0x7d: // ADC abs,X |
320 |
|
|
do_adc(read_byte_abs_x()); |
321 |
|
|
ENDOP(4); |
322 |
|
|
|
323 |
|
|
case 0x79: // ADC abs,Y |
324 |
|
|
do_adc(read_byte_abs_y()); |
325 |
|
|
ENDOP(4); |
326 |
|
|
|
327 |
|
|
case 0x61: // ADC (ind,X) |
328 |
|
|
do_adc(read_byte_ind_x()); |
329 |
|
|
ENDOP(6); |
330 |
|
|
|
331 |
|
|
case 0x71: // ADC (ind),Y |
332 |
|
|
do_adc(read_byte_ind_y()); |
333 |
|
|
ENDOP(5); |
334 |
|
|
|
335 |
|
|
case 0xe9: // SBC #imm |
336 |
|
|
case 0xeb: // Undocumented opcode |
337 |
|
|
do_sbc(read_byte_imm()); |
338 |
|
|
ENDOP(2); |
339 |
|
|
|
340 |
|
|
case 0xe5: // SBC zero |
341 |
|
|
do_sbc(read_byte_zero()); |
342 |
|
|
ENDOP(3); |
343 |
|
|
|
344 |
|
|
case 0xf5: // SBC zero,X |
345 |
|
|
do_sbc(read_byte_zero_x()); |
346 |
|
|
ENDOP(4); |
347 |
|
|
|
348 |
|
|
case 0xed: // SBC abs |
349 |
|
|
do_sbc(read_byte_abs()); |
350 |
|
|
ENDOP(4); |
351 |
|
|
|
352 |
|
|
case 0xfd: // SBC abs,X |
353 |
|
|
do_sbc(read_byte_abs_x()); |
354 |
|
|
ENDOP(4); |
355 |
|
|
|
356 |
|
|
case 0xf9: // SBC abs,Y |
357 |
|
|
do_sbc(read_byte_abs_y()); |
358 |
|
|
ENDOP(4); |
359 |
|
|
|
360 |
|
|
case 0xe1: // SBC (ind,X) |
361 |
|
|
do_sbc(read_byte_ind_x()); |
362 |
|
|
ENDOP(6); |
363 |
|
|
|
364 |
|
|
case 0xf1: // SBC (ind),Y |
365 |
|
|
do_sbc(read_byte_ind_y()); |
366 |
|
|
ENDOP(5); |
367 |
|
|
|
368 |
|
|
|
369 |
|
|
// Increment/decrement group |
370 |
|
|
case 0xe8: // INX |
371 |
|
|
set_nz(++x); |
372 |
|
|
ENDOP(2); |
373 |
|
|
|
374 |
|
|
case 0xca: // DEX |
375 |
|
|
set_nz(--x); |
376 |
|
|
ENDOP(2); |
377 |
|
|
|
378 |
|
|
case 0xc8: // INY |
379 |
|
|
set_nz(++y); |
380 |
|
|
ENDOP(2); |
381 |
|
|
|
382 |
|
|
case 0x88: // DEY |
383 |
|
|
set_nz(--y); |
384 |
|
|
ENDOP(2); |
385 |
|
|
|
386 |
|
|
case 0xe6: // INC zero |
387 |
|
|
adr = read_adr_zero(); |
388 |
|
|
write_zp(adr, set_nz(read_zp(adr) + 1)); |
389 |
|
|
ENDOP(5); |
390 |
|
|
|
391 |
|
|
case 0xf6: // INC zero,X |
392 |
|
|
adr = read_adr_zero_x(); |
393 |
|
|
write_zp(adr, set_nz(read_zp(adr) + 1)); |
394 |
|
|
ENDOP(6); |
395 |
|
|
|
396 |
|
|
case 0xee: // INC abs |
397 |
|
|
adr = read_adr_abs(); |
398 |
|
|
write_byte(adr, set_nz(read_byte(adr) + 1)); |
399 |
|
|
ENDOP(6); |
400 |
|
|
|
401 |
|
|
case 0xfe: // INC abs,X |
402 |
|
|
adr = read_adr_abs_x(); |
403 |
|
|
write_byte(adr, set_nz(read_byte(adr) + 1)); |
404 |
|
|
ENDOP(7); |
405 |
|
|
|
406 |
|
|
case 0xc6: // DEC zero |
407 |
|
|
adr = read_adr_zero(); |
408 |
|
|
write_zp(adr, set_nz(read_zp(adr) - 1)); |
409 |
|
|
ENDOP(5); |
410 |
|
|
|
411 |
|
|
case 0xd6: // DEC zero,X |
412 |
|
|
adr = read_adr_zero_x(); |
413 |
|
|
write_zp(adr, set_nz(read_zp(adr) - 1)); |
414 |
|
|
ENDOP(6); |
415 |
|
|
|
416 |
|
|
case 0xce: // DEC abs |
417 |
|
|
adr = read_adr_abs(); |
418 |
|
|
write_byte(adr, set_nz(read_byte(adr) - 1)); |
419 |
|
|
ENDOP(6); |
420 |
|
|
|
421 |
|
|
case 0xde: // DEC abs,X |
422 |
|
|
adr = read_adr_abs_x(); |
423 |
|
|
write_byte(adr, set_nz(read_byte(adr) - 1)); |
424 |
|
|
ENDOP(7); |
425 |
|
|
|
426 |
|
|
|
427 |
|
|
// Logic group |
428 |
|
|
case 0x29: // AND #imm |
429 |
|
|
set_nz(a &= read_byte_imm()); |
430 |
|
|
ENDOP(2); |
431 |
|
|
|
432 |
|
|
case 0x25: // AND zero |
433 |
|
|
set_nz(a &= read_byte_zero()); |
434 |
|
|
ENDOP(3); |
435 |
|
|
|
436 |
|
|
case 0x35: // AND zero,X |
437 |
|
|
set_nz(a &= read_byte_zero_x()); |
438 |
|
|
ENDOP(4); |
439 |
|
|
|
440 |
|
|
case 0x2d: // AND abs |
441 |
|
|
set_nz(a &= read_byte_abs()); |
442 |
|
|
ENDOP(4); |
443 |
|
|
|
444 |
|
|
case 0x3d: // AND abs,X |
445 |
|
|
set_nz(a &= read_byte_abs_x()); |
446 |
|
|
ENDOP(4); |
447 |
|
|
|
448 |
|
|
case 0x39: // AND abs,Y |
449 |
|
|
set_nz(a &= read_byte_abs_y()); |
450 |
|
|
ENDOP(4); |
451 |
|
|
|
452 |
|
|
case 0x21: // AND (ind,X) |
453 |
|
|
set_nz(a &= read_byte_ind_x()); |
454 |
|
|
ENDOP(6); |
455 |
|
|
|
456 |
|
|
case 0x31: // AND (ind),Y |
457 |
|
|
set_nz(a &= read_byte_ind_y()); |
458 |
|
|
ENDOP(5); |
459 |
|
|
|
460 |
|
|
case 0x09: // ORA #imm |
461 |
|
|
set_nz(a |= read_byte_imm()); |
462 |
|
|
ENDOP(2); |
463 |
|
|
|
464 |
|
|
case 0x05: // ORA zero |
465 |
|
|
set_nz(a |= read_byte_zero()); |
466 |
|
|
ENDOP(3); |
467 |
|
|
|
468 |
|
|
case 0x15: // ORA zero,X |
469 |
|
|
set_nz(a |= read_byte_zero_x()); |
470 |
|
|
ENDOP(4); |
471 |
|
|
|
472 |
|
|
case 0x0d: // ORA abs |
473 |
|
|
set_nz(a |= read_byte_abs()); |
474 |
|
|
ENDOP(4); |
475 |
|
|
|
476 |
|
|
case 0x1d: // ORA abs,X |
477 |
|
|
set_nz(a |= read_byte_abs_x()); |
478 |
|
|
ENDOP(4); |
479 |
|
|
|
480 |
|
|
case 0x19: // ORA abs,Y |
481 |
|
|
set_nz(a |= read_byte_abs_y()); |
482 |
|
|
ENDOP(4); |
483 |
|
|
|
484 |
|
|
case 0x01: // ORA (ind,X) |
485 |
|
|
set_nz(a |= read_byte_ind_x()); |
486 |
|
|
ENDOP(6); |
487 |
|
|
|
488 |
|
|
case 0x11: // ORA (ind),Y |
489 |
|
|
set_nz(a |= read_byte_ind_y()); |
490 |
|
|
ENDOP(5); |
491 |
|
|
|
492 |
|
|
case 0x49: // EOR #imm |
493 |
|
|
set_nz(a ^= read_byte_imm()); |
494 |
|
|
ENDOP(2); |
495 |
|
|
|
496 |
|
|
case 0x45: // EOR zero |
497 |
|
|
set_nz(a ^= read_byte_zero()); |
498 |
|
|
ENDOP(3); |
499 |
|
|
|
500 |
|
|
case 0x55: // EOR zero,X |
501 |
|
|
set_nz(a ^= read_byte_zero_x()); |
502 |
|
|
ENDOP(4); |
503 |
|
|
|
504 |
|
|
case 0x4d: // EOR abs |
505 |
|
|
set_nz(a ^= read_byte_abs()); |
506 |
|
|
ENDOP(4); |
507 |
|
|
|
508 |
|
|
case 0x5d: // EOR abs,X |
509 |
|
|
set_nz(a ^= read_byte_abs_x()); |
510 |
|
|
ENDOP(4); |
511 |
|
|
|
512 |
|
|
case 0x59: // EOR abs,Y |
513 |
|
|
set_nz(a ^= read_byte_abs_y()); |
514 |
|
|
ENDOP(4); |
515 |
|
|
|
516 |
|
|
case 0x41: // EOR (ind,X) |
517 |
|
|
set_nz(a ^= read_byte_ind_x()); |
518 |
|
|
ENDOP(6); |
519 |
|
|
|
520 |
|
|
case 0x51: // EOR (ind),Y |
521 |
|
|
set_nz(a ^= read_byte_ind_y()); |
522 |
|
|
ENDOP(5); |
523 |
|
|
|
524 |
|
|
|
525 |
|
|
// Compare group |
526 |
|
|
case 0xc9: // CMP #imm |
527 |
|
|
set_nz(adr = a - read_byte_imm()); |
528 |
|
|
c_flag = adr < 0x100; |
529 |
|
|
ENDOP(2); |
530 |
|
|
|
531 |
|
|
case 0xc5: // CMP zero |
532 |
|
|
set_nz(adr = a - read_byte_zero()); |
533 |
|
|
c_flag = adr < 0x100; |
534 |
|
|
ENDOP(3); |
535 |
|
|
|
536 |
|
|
case 0xd5: // CMP zero,X |
537 |
|
|
set_nz(adr = a - read_byte_zero_x()); |
538 |
|
|
c_flag = adr < 0x100; |
539 |
|
|
ENDOP(4); |
540 |
|
|
|
541 |
|
|
case 0xcd: // CMP abs |
542 |
|
|
set_nz(adr = a - read_byte_abs()); |
543 |
|
|
c_flag = adr < 0x100; |
544 |
|
|
ENDOP(4); |
545 |
|
|
|
546 |
|
|
case 0xdd: // CMP abs,X |
547 |
|
|
set_nz(adr = a - read_byte_abs_x()); |
548 |
|
|
c_flag = adr < 0x100; |
549 |
|
|
ENDOP(4); |
550 |
|
|
|
551 |
|
|
case 0xd9: // CMP abs,Y |
552 |
|
|
set_nz(adr = a - read_byte_abs_y()); |
553 |
|
|
c_flag = adr < 0x100; |
554 |
|
|
ENDOP(4); |
555 |
|
|
|
556 |
|
|
case 0xc1: // CMP (ind,X) |
557 |
|
|
set_nz(adr = a - read_byte_ind_x()); |
558 |
|
|
c_flag = adr < 0x100; |
559 |
|
|
ENDOP(6); |
560 |
|
|
|
561 |
|
|
case 0xd1: // CMP (ind),Y |
562 |
|
|
set_nz(adr = a - read_byte_ind_y()); |
563 |
|
|
c_flag = adr < 0x100; |
564 |
|
|
ENDOP(5); |
565 |
|
|
|
566 |
|
|
case 0xe0: // CPX #imm |
567 |
|
|
set_nz(adr = x - read_byte_imm()); |
568 |
|
|
c_flag = adr < 0x100; |
569 |
|
|
ENDOP(2); |
570 |
|
|
|
571 |
|
|
case 0xe4: // CPX zero |
572 |
|
|
set_nz(adr = x - read_byte_zero()); |
573 |
|
|
c_flag = adr < 0x100; |
574 |
|
|
ENDOP(3); |
575 |
|
|
|
576 |
|
|
case 0xec: // CPX abs |
577 |
|
|
set_nz(adr = x - read_byte_abs()); |
578 |
|
|
c_flag = adr < 0x100; |
579 |
|
|
ENDOP(4); |
580 |
|
|
|
581 |
|
|
case 0xc0: // CPY #imm |
582 |
|
|
set_nz(adr = y - read_byte_imm()); |
583 |
|
|
c_flag = adr < 0x100; |
584 |
|
|
ENDOP(2); |
585 |
|
|
|
586 |
|
|
case 0xc4: // CPY zero |
587 |
|
|
set_nz(adr = y - read_byte_zero()); |
588 |
|
|
c_flag = adr < 0x100; |
589 |
|
|
ENDOP(3); |
590 |
|
|
|
591 |
|
|
case 0xcc: // CPY abs |
592 |
|
|
set_nz(adr = y - read_byte_abs()); |
593 |
|
|
c_flag = adr < 0x100; |
594 |
|
|
ENDOP(4); |
595 |
|
|
|
596 |
|
|
|
597 |
|
|
// Bit-test group |
598 |
|
|
case 0x24: // BIT zero |
599 |
|
|
z_flag = a & (tmp = read_byte_zero()); |
600 |
|
|
n_flag = tmp; |
601 |
|
|
v_flag = tmp & 0x40; |
602 |
|
|
ENDOP(3); |
603 |
|
|
|
604 |
|
|
case 0x2c: // BIT abs |
605 |
|
|
z_flag = a & (tmp = read_byte_abs()); |
606 |
|
|
n_flag = tmp; |
607 |
|
|
v_flag = tmp & 0x40; |
608 |
|
|
ENDOP(4); |
609 |
|
|
|
610 |
|
|
|
611 |
|
|
// Shift/rotate group |
612 |
|
|
case 0x0a: // ASL A |
613 |
|
|
c_flag = a & 0x80; |
614 |
|
|
set_nz(a <<= 1); |
615 |
|
|
ENDOP(2); |
616 |
|
|
|
617 |
|
|
case 0x06: // ASL zero |
618 |
|
|
tmp = read_zp(adr = read_adr_zero()); |
619 |
|
|
c_flag = tmp & 0x80; |
620 |
|
|
write_zp(adr, set_nz(tmp << 1)); |
621 |
|
|
ENDOP(5); |
622 |
|
|
|
623 |
|
|
case 0x16: // ASL zero,X |
624 |
|
|
tmp = read_zp(adr = read_adr_zero_x()); |
625 |
|
|
c_flag = tmp & 0x80; |
626 |
|
|
write_zp(adr, set_nz(tmp << 1)); |
627 |
|
|
ENDOP(6); |
628 |
|
|
|
629 |
|
|
case 0x0e: // ASL abs |
630 |
|
|
tmp = read_byte(adr = read_adr_abs()); |
631 |
|
|
c_flag = tmp & 0x80; |
632 |
|
|
write_byte(adr, set_nz(tmp << 1)); |
633 |
|
|
ENDOP(6); |
634 |
|
|
|
635 |
|
|
case 0x1e: // ASL abs,X |
636 |
|
|
tmp = read_byte(adr = read_adr_abs_x()); |
637 |
|
|
c_flag = tmp & 0x80; |
638 |
|
|
write_byte(adr, set_nz(tmp << 1)); |
639 |
|
|
ENDOP(7); |
640 |
|
|
|
641 |
|
|
case 0x4a: // LSR A |
642 |
|
|
c_flag = a & 0x01; |
643 |
|
|
set_nz(a >>= 1); |
644 |
|
|
ENDOP(2); |
645 |
|
|
|
646 |
|
|
case 0x46: // LSR zero |
647 |
|
|
tmp = read_zp(adr = read_adr_zero()); |
648 |
|
|
c_flag = tmp & 0x01; |
649 |
|
|
write_zp(adr, set_nz(tmp >> 1)); |
650 |
|
|
ENDOP(5); |
651 |
|
|
|
652 |
|
|
case 0x56: // LSR zero,X |
653 |
|
|
tmp = read_zp(adr = read_adr_zero_x()); |
654 |
|
|
c_flag = tmp & 0x01; |
655 |
|
|
write_zp(adr, set_nz(tmp >> 1)); |
656 |
|
|
ENDOP(6); |
657 |
|
|
|
658 |
|
|
case 0x4e: // LSR abs |
659 |
|
|
tmp = read_byte(adr = read_adr_abs()); |
660 |
|
|
c_flag = tmp & 0x01; |
661 |
|
|
write_byte(adr, set_nz(tmp >> 1)); |
662 |
|
|
ENDOP(6); |
663 |
|
|
|
664 |
|
|
case 0x5e: // LSR abs,X |
665 |
|
|
tmp = read_byte(adr = read_adr_abs_x()); |
666 |
|
|
c_flag = tmp & 0x01; |
667 |
|
|
write_byte(adr, set_nz(tmp >> 1)); |
668 |
|
|
ENDOP(7); |
669 |
|
|
|
670 |
|
|
case 0x2a: // ROL A |
671 |
|
|
tmp2 = a & 0x80; |
672 |
|
|
set_nz(a = c_flag ? (a << 1) | 0x01 : a << 1); |
673 |
|
|
c_flag = tmp2; |
674 |
|
|
ENDOP(2); |
675 |
|
|
|
676 |
|
|
case 0x26: // ROL zero |
677 |
|
|
tmp = read_zp(adr = read_adr_zero()); |
678 |
|
|
tmp2 = tmp & 0x80; |
679 |
|
|
write_zp(adr, set_nz(c_flag ? (tmp << 1) | 0x01 : tmp << 1)); |
680 |
|
|
c_flag = tmp2; |
681 |
|
|
ENDOP(5); |
682 |
|
|
|
683 |
|
|
case 0x36: // ROL zero,X |
684 |
|
|
tmp = read_zp(adr = read_adr_zero_x()); |
685 |
|
|
tmp2 = tmp & 0x80; |
686 |
|
|
write_zp(adr, set_nz(c_flag ? (tmp << 1) | 0x01 : tmp << 1)); |
687 |
|
|
c_flag = tmp2; |
688 |
|
|
ENDOP(6); |
689 |
|
|
|
690 |
|
|
case 0x2e: // ROL abs |
691 |
|
|
tmp = read_byte(adr = read_adr_abs()); |
692 |
|
|
tmp2 = tmp & 0x80; |
693 |
|
|
write_byte(adr, set_nz(c_flag ? (tmp << 1) | 0x01 : tmp << 1)); |
694 |
|
|
c_flag = tmp2; |
695 |
|
|
ENDOP(6); |
696 |
|
|
|
697 |
|
|
case 0x3e: // ROL abs,X |
698 |
|
|
tmp = read_byte(adr = read_adr_abs_x()); |
699 |
|
|
tmp2 = tmp & 0x80; |
700 |
|
|
write_byte(adr, set_nz(c_flag ? (tmp << 1) | 0x01 : tmp << 1)); |
701 |
|
|
c_flag = tmp2; |
702 |
|
|
ENDOP(7); |
703 |
|
|
|
704 |
|
|
case 0x6a: // ROR A |
705 |
|
|
tmp2 = a & 0x01; |
706 |
|
|
set_nz(a = (c_flag ? (a >> 1) | 0x80 : a >> 1)); |
707 |
|
|
c_flag = tmp2; |
708 |
|
|
ENDOP(2); |
709 |
|
|
|
710 |
|
|
case 0x66: // ROR zero |
711 |
|
|
tmp = read_zp(adr = read_adr_zero()); |
712 |
|
|
tmp2 = tmp & 0x01; |
713 |
|
|
write_zp(adr, set_nz(c_flag ? (tmp >> 1) | 0x80 : tmp >> 1)); |
714 |
|
|
c_flag = tmp2; |
715 |
|
|
ENDOP(5); |
716 |
|
|
|
717 |
|
|
case 0x76: // ROR zero,X |
718 |
|
|
tmp = read_zp(adr = read_adr_zero_x()); |
719 |
|
|
tmp2 = tmp & 0x01; |
720 |
|
|
write_zp(adr, set_nz(c_flag ? (tmp >> 1) | 0x80 : tmp >> 1)); |
721 |
|
|
c_flag = tmp2; |
722 |
|
|
ENDOP(6); |
723 |
|
|
|
724 |
|
|
case 0x6e: // ROR abs |
725 |
|
|
tmp = read_byte(adr = read_adr_abs()); |
726 |
|
|
tmp2 = tmp & 0x01; |
727 |
|
|
write_byte(adr, set_nz(c_flag ? (tmp >> 1) | 0x80 : tmp >> 1)); |
728 |
|
|
c_flag = tmp2; |
729 |
|
|
ENDOP(6); |
730 |
|
|
|
731 |
|
|
case 0x7e: // ROR abs,X |
732 |
|
|
tmp = read_byte(adr = read_adr_abs_x()); |
733 |
|
|
tmp2 = tmp & 0x01; |
734 |
|
|
write_byte(adr, set_nz(c_flag ? (tmp >> 1) | 0x80 : tmp >> 1)); |
735 |
|
|
c_flag = tmp2; |
736 |
|
|
ENDOP(7); |
737 |
|
|
|
738 |
|
|
|
739 |
|
|
// Stack group |
740 |
|
|
case 0x48: // PHA |
741 |
|
|
push_byte(a); |
742 |
|
|
ENDOP(3); |
743 |
|
|
|
744 |
|
|
case 0x68: // PLA |
745 |
|
|
set_nz(a = pop_byte()); |
746 |
|
|
ENDOP(4); |
747 |
|
|
|
748 |
|
|
case 0x08: // PHP |
749 |
|
|
push_flags(true); |
750 |
|
|
ENDOP(3); |
751 |
|
|
|
752 |
|
|
case 0x28: // PLP |
753 |
|
|
pop_flags(); |
754 |
|
|
if (interrupt.intr_any && !i_flag) |
755 |
|
|
goto handle_int; |
756 |
|
|
ENDOP(4); |
757 |
|
|
|
758 |
|
|
|
759 |
|
|
// Jump/branch group |
760 |
|
|
case 0x4c: // JMP abs |
761 |
|
|
jump(read_adr_abs()); |
762 |
|
|
ENDOP(3); |
763 |
|
|
|
764 |
|
|
case 0x6c: // JMP (ind) |
765 |
|
|
adr = read_adr_abs(); |
766 |
|
|
jump(read_byte(adr) | (read_byte((adr + 1) & 0xff | adr & 0xff00) << 8)); |
767 |
|
|
ENDOP(5); |
768 |
|
|
|
769 |
|
|
case 0x20: // JSR abs |
770 |
|
|
#if PC_IS_POINTER |
771 |
|
|
push_byte((pc-pc_base+1) >> 8); push_byte(pc-pc_base+1); |
772 |
|
|
#else |
773 |
|
|
push_byte(pc+1 >> 8); push_byte(pc+1); |
774 |
|
|
#endif |
775 |
|
|
jump(read_adr_abs()); |
776 |
|
|
ENDOP(6); |
777 |
|
|
|
778 |
|
|
case 0x60: // RTS |
779 |
|
|
adr = pop_byte(); // Split because of pop_byte ++sp side-effect |
780 |
|
|
jump((adr | pop_byte() << 8) + 1); |
781 |
|
|
ENDOP(6); |
782 |
|
|
|
783 |
|
|
case 0x40: // RTI |
784 |
|
|
pop_flags(); |
785 |
|
|
adr = pop_byte(); // Split because of pop_byte ++sp side-effect |
786 |
|
|
jump(adr | pop_byte() << 8); |
787 |
|
|
if (interrupt.intr_any && !i_flag) |
788 |
|
|
goto handle_int; |
789 |
|
|
ENDOP(6); |
790 |
|
|
|
791 |
|
|
case 0x00: // BRK |
792 |
|
|
#if PC_IS_POINTER |
793 |
|
|
push_byte((pc+1-pc_base) >> 8); push_byte(pc+1-pc_base); |
794 |
|
|
#else |
795 |
|
|
push_byte((pc+1) >> 8); push_byte(pc+1); |
796 |
|
|
#endif |
797 |
|
|
push_flags(true); |
798 |
|
|
i_flag = true; |
799 |
|
|
jump(read_word(0xfffe)); |
800 |
|
|
ENDOP(7); |
801 |
|
|
|
802 |
|
|
#if PC_IS_POINTER |
803 |
|
|
#if PRECISE_CPU_CYCLES |
804 |
|
|
#define Branch(flag) \ |
805 |
|
|
if (flag) { \ |
806 |
|
|
pc += (int8)*pc + 1; \ |
807 |
|
|
if (((pc-pc_base) ^ (old_pc - pc_base)) & 0xff00) { \ |
808 |
|
|
ENDOP(4); \ |
809 |
|
|
} else { \ |
810 |
|
|
ENDOP(3); \ |
811 |
|
|
} \ |
812 |
|
|
} else { \ |
813 |
|
|
pc++; \ |
814 |
|
|
ENDOP(2); \ |
815 |
|
|
} |
816 |
|
|
#else |
817 |
|
|
#define Branch(flag) \ |
818 |
|
|
if (flag) { \ |
819 |
|
|
pc += (int8)*pc + 1; \ |
820 |
|
|
ENDOP(3); \ |
821 |
|
|
} else { \ |
822 |
|
|
pc++; \ |
823 |
|
|
ENDOP(2); \ |
824 |
|
|
} |
825 |
|
|
#endif |
826 |
|
|
#else |
827 |
|
|
#define Branch(flag) \ |
828 |
|
|
if (flag) { \ |
829 |
|
|
uint16 old_pc = pc; \ |
830 |
|
|
pc += (int8)read_byte(pc) + 1; \ |
831 |
|
|
if ((pc ^ old_pc) & 0xff00) { \ |
832 |
|
|
ENDOP(4); \ |
833 |
|
|
} else { \ |
834 |
|
|
ENDOP(3); \ |
835 |
|
|
} \ |
836 |
|
|
} else { \ |
837 |
|
|
pc++; \ |
838 |
|
|
ENDOP(2); \ |
839 |
|
|
} |
840 |
|
|
#endif |
841 |
|
|
|
842 |
|
|
case 0xb0: // BCS rel |
843 |
|
|
Branch(c_flag); |
844 |
|
|
|
845 |
|
|
case 0x90: // BCC rel |
846 |
|
|
Branch(!c_flag); |
847 |
|
|
|
848 |
|
|
case 0xf0: // BEQ rel |
849 |
|
|
Branch(!z_flag); |
850 |
|
|
|
851 |
|
|
case 0xd0: // BNE rel |
852 |
|
|
Branch(z_flag); |
853 |
|
|
|
854 |
|
|
case 0x70: // BVS rel |
855 |
|
|
#ifndef IS_CPU_1541 |
856 |
|
|
Branch(v_flag); |
857 |
|
|
#else |
858 |
|
|
Branch((via2_pcr & 0x0e) == 0x0e ? 1 : v_flag); // GCR byte ready flag |
859 |
|
|
#endif |
860 |
|
|
|
861 |
|
|
case 0x50: // BVC rel |
862 |
|
|
#ifndef IS_CPU_1541 |
863 |
|
|
Branch(!v_flag); |
864 |
|
|
#else |
865 |
|
|
Branch(!((via2_pcr & 0x0e) == 0x0e) ? 0 : v_flag); // GCR byte ready flag |
866 |
|
|
#endif |
867 |
|
|
|
868 |
|
|
case 0x30: // BMI rel |
869 |
|
|
Branch(n_flag & 0x80); |
870 |
|
|
|
871 |
|
|
case 0x10: // BPL rel |
872 |
|
|
Branch(!(n_flag & 0x80)); |
873 |
|
|
|
874 |
|
|
|
875 |
|
|
// Flags group |
876 |
|
|
case 0x38: // SEC |
877 |
|
|
c_flag = true; |
878 |
|
|
ENDOP(2); |
879 |
|
|
|
880 |
|
|
case 0x18: // CLC |
881 |
|
|
c_flag = false; |
882 |
|
|
ENDOP(2); |
883 |
|
|
|
884 |
|
|
case 0xf8: // SED |
885 |
|
|
d_flag = true; |
886 |
|
|
ENDOP(2); |
887 |
|
|
|
888 |
|
|
case 0xd8: // CLD |
889 |
|
|
d_flag = false; |
890 |
|
|
ENDOP(2); |
891 |
|
|
|
892 |
|
|
case 0x78: // SEI |
893 |
|
|
i_flag = true; |
894 |
|
|
ENDOP(2); |
895 |
|
|
|
896 |
|
|
case 0x58: // CLI |
897 |
|
|
i_flag = false; |
898 |
|
|
if (interrupt.intr_any) |
899 |
|
|
goto handle_int; |
900 |
|
|
ENDOP(2); |
901 |
|
|
|
902 |
|
|
case 0xb8: // CLV |
903 |
|
|
v_flag = false; |
904 |
|
|
ENDOP(2); |
905 |
|
|
|
906 |
|
|
|
907 |
|
|
// NOP group |
908 |
|
|
case 0xea: // NOP |
909 |
|
|
ENDOP(2); |
910 |
|
|
|
911 |
|
|
|
912 |
|
|
/* |
913 |
|
|
* Undocumented opcodes start here |
914 |
|
|
*/ |
915 |
|
|
|
916 |
|
|
// NOP group |
917 |
|
|
case 0x1a: // NOP |
918 |
|
|
case 0x3a: |
919 |
|
|
case 0x5a: |
920 |
|
|
case 0x7a: |
921 |
|
|
case 0xda: |
922 |
|
|
case 0xfa: |
923 |
|
|
ENDOP(2); |
924 |
|
|
|
925 |
|
|
case 0x80: // NOP #imm |
926 |
|
|
case 0x82: |
927 |
|
|
case 0x89: |
928 |
|
|
case 0xc2: |
929 |
|
|
case 0xe2: |
930 |
|
|
pc++; |
931 |
|
|
ENDOP(2); |
932 |
|
|
|
933 |
|
|
case 0x04: // NOP zero |
934 |
|
|
case 0x44: |
935 |
|
|
case 0x64: |
936 |
|
|
pc++; |
937 |
|
|
ENDOP(3); |
938 |
|
|
|
939 |
|
|
case 0x14: // NOP zero,X |
940 |
|
|
case 0x34: |
941 |
|
|
case 0x54: |
942 |
|
|
case 0x74: |
943 |
|
|
case 0xd4: |
944 |
|
|
case 0xf4: |
945 |
|
|
pc++; |
946 |
|
|
ENDOP(4); |
947 |
|
|
|
948 |
|
|
case 0x0c: // NOP abs |
949 |
|
|
pc+=2; |
950 |
|
|
ENDOP(4); |
951 |
|
|
|
952 |
|
|
case 0x1c: // NOP abs,X |
953 |
|
|
case 0x3c: |
954 |
|
|
case 0x5c: |
955 |
|
|
case 0x7c: |
956 |
|
|
case 0xdc: |
957 |
|
|
case 0xfc: |
958 |
|
|
#if PRECISE_CPU_CYCLES |
959 |
|
|
read_byte_abs_x(); |
960 |
|
|
#else |
961 |
|
|
pc+=2; |
962 |
|
|
#endif |
963 |
|
|
ENDOP(4); |
964 |
|
|
|
965 |
|
|
|
966 |
|
|
// Load A/X group |
967 |
|
|
case 0xa7: // LAX zero |
968 |
|
|
set_nz(a = x = read_byte_zero()); |
969 |
|
|
ENDOP(3); |
970 |
|
|
|
971 |
|
|
case 0xb7: // LAX zero,Y |
972 |
|
|
set_nz(a = x = read_byte_zero_y()); |
973 |
|
|
ENDOP(4); |
974 |
|
|
|
975 |
|
|
case 0xaf: // LAX abs |
976 |
|
|
set_nz(a = x = read_byte_abs()); |
977 |
|
|
ENDOP(4); |
978 |
|
|
|
979 |
|
|
case 0xbf: // LAX abs,Y |
980 |
|
|
set_nz(a = x = read_byte_abs_y()); |
981 |
|
|
ENDOP(4); |
982 |
|
|
|
983 |
|
|
case 0xa3: // LAX (ind,X) |
984 |
|
|
set_nz(a = x = read_byte_ind_x()); |
985 |
|
|
ENDOP(6); |
986 |
|
|
|
987 |
|
|
case 0xb3: // LAX (ind),Y |
988 |
|
|
set_nz(a = x = read_byte_ind_y()); |
989 |
|
|
ENDOP(5); |
990 |
|
|
|
991 |
|
|
|
992 |
|
|
// Store A/X group |
993 |
|
|
case 0x87: // SAX zero |
994 |
|
|
write_byte(read_adr_zero(), a & x); |
995 |
|
|
ENDOP(3); |
996 |
|
|
|
997 |
|
|
case 0x97: // SAX zero,Y |
998 |
|
|
write_byte(read_adr_zero_y(), a & x); |
999 |
|
|
ENDOP(4); |
1000 |
|
|
|
1001 |
|
|
case 0x8f: // SAX abs |
1002 |
|
|
write_byte(read_adr_abs(), a & x); |
1003 |
|
|
ENDOP(4); |
1004 |
|
|
|
1005 |
|
|
case 0x83: // SAX (ind,X) |
1006 |
|
|
write_byte(read_adr_ind_x(), a & x); |
1007 |
|
|
ENDOP(6); |
1008 |
|
|
|
1009 |
|
|
|
1010 |
|
|
// ASL/ORA group |
1011 |
|
|
#define ShiftLeftOr \ |
1012 |
|
|
c_flag = tmp & 0x80; \ |
1013 |
|
|
tmp <<= 1; \ |
1014 |
|
|
set_nz(a |= tmp); |
1015 |
|
|
|
1016 |
|
|
case 0x07: // SLO zero |
1017 |
|
|
tmp = read_zp(adr = read_adr_zero()); |
1018 |
|
|
ShiftLeftOr; |
1019 |
|
|
write_zp(adr, tmp); |
1020 |
|
|
ENDOP(5); |
1021 |
|
|
|
1022 |
|
|
case 0x17: // SLO zero,X |
1023 |
|
|
tmp = read_zp(adr = read_adr_zero_x()); |
1024 |
|
|
ShiftLeftOr; |
1025 |
|
|
write_zp(adr, tmp); |
1026 |
|
|
ENDOP(6); |
1027 |
|
|
|
1028 |
|
|
case 0x0f: // SLO abs |
1029 |
|
|
tmp = read_byte(adr = read_adr_abs()); |
1030 |
|
|
ShiftLeftOr; |
1031 |
|
|
write_byte(adr, tmp); |
1032 |
|
|
ENDOP(6); |
1033 |
|
|
|
1034 |
|
|
case 0x1f: // SLO abs,X |
1035 |
|
|
tmp = read_byte(adr = read_adr_abs_x()); |
1036 |
|
|
ShiftLeftOr; |
1037 |
|
|
write_byte(adr, tmp); |
1038 |
|
|
ENDOP(7); |
1039 |
|
|
|
1040 |
|
|
case 0x1b: // SLO abs,Y |
1041 |
|
|
tmp = read_byte(adr = read_adr_abs_y()); |
1042 |
|
|
ShiftLeftOr; |
1043 |
|
|
write_byte(adr, tmp); |
1044 |
|
|
ENDOP(7); |
1045 |
|
|
|
1046 |
|
|
case 0x03: // SLO (ind,X) |
1047 |
|
|
tmp = read_byte(adr = read_adr_ind_x()); |
1048 |
|
|
ShiftLeftOr; |
1049 |
|
|
write_byte(adr, tmp); |
1050 |
|
|
ENDOP(8); |
1051 |
|
|
|
1052 |
|
|
case 0x13: // SLO (ind),Y |
1053 |
|
|
tmp = read_byte(adr = read_adr_ind_y()); |
1054 |
|
|
ShiftLeftOr; |
1055 |
|
|
write_byte(adr, tmp); |
1056 |
|
|
ENDOP(8); |
1057 |
|
|
|
1058 |
|
|
|
1059 |
|
|
// ROL/AND group |
1060 |
|
|
#define RoLeftAnd \ |
1061 |
|
|
tmp2 = tmp & 0x80; \ |
1062 |
|
|
tmp = c_flag ? (tmp << 1) | 0x01 : tmp << 1; \ |
1063 |
|
|
set_nz(a &= tmp); \ |
1064 |
|
|
c_flag = tmp2; |
1065 |
|
|
|
1066 |
|
|
case 0x27: // RLA zero |
1067 |
|
|
tmp = read_zp(adr = read_adr_zero()); |
1068 |
|
|
RoLeftAnd; |
1069 |
|
|
write_zp(adr, tmp); |
1070 |
|
|
ENDOP(5); |
1071 |
|
|
|
1072 |
|
|
case 0x37: // RLA zero,X |
1073 |
|
|
tmp = read_zp(adr = read_adr_zero_x()); |
1074 |
|
|
RoLeftAnd; |
1075 |
|
|
write_zp(adr, tmp); |
1076 |
|
|
ENDOP(6); |
1077 |
|
|
|
1078 |
|
|
case 0x2f: // RLA abs |
1079 |
|
|
tmp = read_byte(adr = read_adr_abs()); |
1080 |
|
|
RoLeftAnd; |
1081 |
|
|
write_byte(adr, tmp); |
1082 |
|
|
ENDOP(6); |
1083 |
|
|
|
1084 |
|
|
case 0x3f: // RLA abs,X |
1085 |
|
|
tmp = read_byte(adr = read_adr_abs_x()); |
1086 |
|
|
RoLeftAnd; |
1087 |
|
|
write_byte(adr, tmp); |
1088 |
|
|
ENDOP(7); |
1089 |
|
|
|
1090 |
|
|
case 0x3b: // RLA abs,Y |
1091 |
|
|
tmp = read_byte(adr = read_adr_abs_y()); |
1092 |
|
|
RoLeftAnd; |
1093 |
|
|
write_byte(adr, tmp); |
1094 |
|
|
ENDOP(7); |
1095 |
|
|
|
1096 |
|
|
case 0x23: // RLA (ind,X) |
1097 |
|
|
tmp = read_byte(adr = read_adr_ind_x()); |
1098 |
|
|
RoLeftAnd; |
1099 |
|
|
write_byte(adr, tmp); |
1100 |
|
|
ENDOP(8); |
1101 |
|
|
|
1102 |
|
|
case 0x33: // RLA (ind),Y |
1103 |
|
|
tmp = read_byte(adr = read_adr_ind_y()); |
1104 |
|
|
RoLeftAnd; |
1105 |
|
|
write_byte(adr, tmp); |
1106 |
|
|
ENDOP(8); |
1107 |
|
|
|
1108 |
|
|
|
1109 |
|
|
// LSR/EOR group |
1110 |
|
|
#define ShiftRightEor \ |
1111 |
|
|
c_flag = tmp & 0x01; \ |
1112 |
|
|
tmp >>= 1; \ |
1113 |
|
|
set_nz(a ^= tmp); |
1114 |
|
|
|
1115 |
|
|
case 0x47: // SRE zero |
1116 |
|
|
tmp = read_zp(adr = read_adr_zero()); |
1117 |
|
|
ShiftRightEor; |
1118 |
|
|
write_zp(adr, tmp); |
1119 |
|
|
ENDOP(5); |
1120 |
|
|
|
1121 |
|
|
case 0x57: // SRE zero,X |
1122 |
|
|
tmp = read_zp(adr = read_adr_zero_x()); |
1123 |
|
|
ShiftRightEor; |
1124 |
|
|
write_zp(adr, tmp); |
1125 |
|
|
ENDOP(6); |
1126 |
|
|
|
1127 |
|
|
case 0x4f: // SRE abs |
1128 |
|
|
tmp = read_byte(adr = read_adr_abs()); |
1129 |
|
|
ShiftRightEor; |
1130 |
|
|
write_byte(adr, tmp); |
1131 |
|
|
ENDOP(6); |
1132 |
|
|
|
1133 |
|
|
case 0x5f: // SRE abs,X |
1134 |
|
|
tmp = read_byte(adr = read_adr_abs_x()); |
1135 |
|
|
ShiftRightEor; |
1136 |
|
|
write_byte(adr, tmp); |
1137 |
|
|
ENDOP(7); |
1138 |
|
|
|
1139 |
|
|
case 0x5b: // SRE abs,Y |
1140 |
|
|
tmp = read_byte(adr = read_adr_abs_y()); |
1141 |
|
|
ShiftRightEor; |
1142 |
|
|
write_byte(adr, tmp); |
1143 |
|
|
ENDOP(7); |
1144 |
|
|
|
1145 |
|
|
case 0x43: // SRE (ind,X) |
1146 |
|
|
tmp = read_byte(adr = read_adr_ind_x()); |
1147 |
|
|
ShiftRightEor; |
1148 |
|
|
write_byte(adr, tmp); |
1149 |
|
|
ENDOP(8); |
1150 |
|
|
|
1151 |
|
|
case 0x53: // SRE (ind),Y |
1152 |
|
|
tmp = read_byte(adr = read_adr_ind_y()); |
1153 |
|
|
ShiftRightEor; |
1154 |
|
|
write_byte(adr, tmp); |
1155 |
|
|
ENDOP(8); |
1156 |
|
|
|
1157 |
|
|
|
1158 |
|
|
// ROR/ADC group |
1159 |
|
|
#define RoRightAdc \ |
1160 |
|
|
tmp2 = tmp & 0x01; \ |
1161 |
|
|
tmp = c_flag ? (tmp >> 1) | 0x80 : tmp >> 1; \ |
1162 |
|
|
c_flag = tmp2; \ |
1163 |
|
|
do_adc(tmp); |
1164 |
|
|
|
1165 |
|
|
case 0x67: // RRA zero |
1166 |
|
|
tmp = read_zp(adr = read_adr_zero()); |
1167 |
|
|
RoRightAdc; |
1168 |
|
|
write_zp(adr, tmp); |
1169 |
|
|
ENDOP(5); |
1170 |
|
|
|
1171 |
|
|
case 0x77: // RRA zero,X |
1172 |
|
|
tmp = read_zp(adr = read_adr_zero_x()); |
1173 |
|
|
RoRightAdc; |
1174 |
|
|
write_zp(adr, tmp); |
1175 |
|
|
ENDOP(6); |
1176 |
|
|
|
1177 |
|
|
case 0x6f: // RRA abs |
1178 |
|
|
tmp = read_byte(adr = read_adr_abs()); |
1179 |
|
|
RoRightAdc; |
1180 |
|
|
write_byte(adr, tmp); |
1181 |
|
|
ENDOP(6); |
1182 |
|
|
|
1183 |
|
|
case 0x7f: // RRA abs,X |
1184 |
|
|
tmp = read_byte(adr = read_adr_abs_x()); |
1185 |
|
|
RoRightAdc; |
1186 |
|
|
write_byte(adr, tmp); |
1187 |
|
|
ENDOP(7); |
1188 |
|
|
|
1189 |
|
|
case 0x7b: // RRA abs,Y |
1190 |
|
|
tmp = read_byte(adr = read_adr_abs_y()); |
1191 |
|
|
RoRightAdc; |
1192 |
|
|
write_byte(adr, tmp); |
1193 |
|
|
ENDOP(7); |
1194 |
|
|
|
1195 |
|
|
case 0x63: // RRA (ind,X) |
1196 |
|
|
tmp = read_byte(adr = read_adr_ind_x()); |
1197 |
|
|
RoRightAdc; |
1198 |
|
|
write_byte(adr, tmp); |
1199 |
|
|
ENDOP(8); |
1200 |
|
|
|
1201 |
|
|
case 0x73: // RRA (ind),Y |
1202 |
|
|
tmp = read_byte(adr = read_adr_ind_y()); |
1203 |
|
|
RoRightAdc; |
1204 |
|
|
write_byte(adr, tmp); |
1205 |
|
|
ENDOP(8); |
1206 |
|
|
|
1207 |
|
|
|
1208 |
|
|
// DEC/CMP group |
1209 |
|
|
#define DecCompare \ |
1210 |
|
|
set_nz(adr = a - tmp); \ |
1211 |
|
|
c_flag = adr < 0x100; |
1212 |
|
|
|
1213 |
|
|
case 0xc7: // DCP zero |
1214 |
|
|
tmp = read_zp(adr = read_adr_zero()) - 1; |
1215 |
|
|
write_zp(adr, tmp); |
1216 |
|
|
DecCompare; |
1217 |
|
|
ENDOP(5); |
1218 |
|
|
|
1219 |
|
|
case 0xd7: // DCP zero,X |
1220 |
|
|
tmp = read_zp(adr = read_adr_zero_x()) - 1; |
1221 |
|
|
write_zp(adr, tmp); |
1222 |
|
|
DecCompare; |
1223 |
|
|
ENDOP(6); |
1224 |
|
|
|
1225 |
|
|
case 0xcf: // DCP abs |
1226 |
|
|
tmp = read_byte(adr = read_adr_abs()) - 1; |
1227 |
|
|
write_byte(adr, tmp); |
1228 |
|
|
DecCompare; |
1229 |
|
|
ENDOP(6); |
1230 |
|
|
|
1231 |
|
|
case 0xdf: // DCP abs,X |
1232 |
|
|
tmp = read_byte(adr = read_adr_abs_x()) - 1; |
1233 |
|
|
write_byte(adr, tmp); |
1234 |
|
|
DecCompare; |
1235 |
|
|
ENDOP(7); |
1236 |
|
|
|
1237 |
|
|
case 0xdb: // DCP abs,Y |
1238 |
|
|
tmp = read_byte(adr = read_adr_abs_y()) - 1; |
1239 |
|
|
write_byte(adr, tmp); |
1240 |
|
|
DecCompare; |
1241 |
|
|
ENDOP(7); |
1242 |
|
|
|
1243 |
|
|
case 0xc3: // DCP (ind,X) |
1244 |
|
|
tmp = read_byte(adr = read_adr_ind_x()) - 1; |
1245 |
|
|
write_byte(adr, tmp); |
1246 |
|
|
DecCompare; |
1247 |
|
|
ENDOP(8); |
1248 |
|
|
|
1249 |
|
|
case 0xd3: // DCP (ind),Y |
1250 |
|
|
tmp = read_byte(adr = read_adr_ind_y()) - 1; |
1251 |
|
|
write_byte(adr, tmp); |
1252 |
|
|
DecCompare; |
1253 |
|
|
ENDOP(8); |
1254 |
|
|
|
1255 |
|
|
|
1256 |
|
|
// INC/SBC group |
1257 |
|
|
case 0xe7: // ISB zero |
1258 |
|
|
tmp = read_zp(adr = read_adr_zero()) + 1; |
1259 |
|
|
do_sbc(tmp); |
1260 |
|
|
write_zp(adr, tmp); |
1261 |
|
|
ENDOP(5); |
1262 |
|
|
|
1263 |
|
|
case 0xf7: // ISB zero,X |
1264 |
|
|
tmp = read_zp(adr = read_adr_zero_x()) + 1; |
1265 |
|
|
do_sbc(tmp); |
1266 |
|
|
write_zp(adr, tmp); |
1267 |
|
|
ENDOP(6); |
1268 |
|
|
|
1269 |
|
|
case 0xef: // ISB abs |
1270 |
|
|
tmp = read_byte(adr = read_adr_abs()) + 1; |
1271 |
|
|
do_sbc(tmp); |
1272 |
|
|
write_byte(adr, tmp); |
1273 |
|
|
ENDOP(6); |
1274 |
|
|
|
1275 |
|
|
case 0xff: // ISB abs,X |
1276 |
|
|
tmp = read_byte(adr = read_adr_abs_x()) + 1; |
1277 |
|
|
do_sbc(tmp); |
1278 |
|
|
write_byte(adr, tmp); |
1279 |
|
|
ENDOP(7); |
1280 |
|
|
|
1281 |
|
|
case 0xfb: // ISB abs,Y |
1282 |
|
|
tmp = read_byte(adr = read_adr_abs_y()) + 1; |
1283 |
|
|
do_sbc(tmp); |
1284 |
|
|
write_byte(adr, tmp); |
1285 |
|
|
ENDOP(7); |
1286 |
|
|
|
1287 |
|
|
case 0xe3: // ISB (ind,X) |
1288 |
|
|
tmp = read_byte(adr = read_adr_ind_x()) + 1; |
1289 |
|
|
do_sbc(tmp); |
1290 |
|
|
write_byte(adr, tmp); |
1291 |
|
|
ENDOP(8); |
1292 |
|
|
|
1293 |
|
|
case 0xf3: // ISB (ind),Y |
1294 |
|
|
tmp = read_byte(adr = read_adr_ind_y()) + 1; |
1295 |
|
|
do_sbc(tmp); |
1296 |
|
|
write_byte(adr, tmp); |
1297 |
|
|
ENDOP(8); |
1298 |
|
|
|
1299 |
|
|
|
1300 |
|
|
// Complex functions |
1301 |
|
|
case 0x0b: // ANC #imm |
1302 |
|
|
case 0x2b: |
1303 |
|
|
set_nz(a &= read_byte_imm()); |
1304 |
|
|
c_flag = n_flag & 0x80; |
1305 |
|
|
ENDOP(2); |
1306 |
|
|
|
1307 |
|
|
case 0x4b: // ASR #imm |
1308 |
|
|
a &= read_byte_imm(); |
1309 |
|
|
c_flag = a & 0x01; |
1310 |
|
|
set_nz(a >>= 1); |
1311 |
|
|
ENDOP(2); |
1312 |
|
|
|
1313 |
|
|
case 0x6b: // ARR #imm |
1314 |
|
|
tmp2 = read_byte_imm() & a; |
1315 |
|
|
a = (c_flag ? (tmp2 >> 1) | 0x80 : tmp2 >> 1); |
1316 |
|
|
if (!d_flag) { |
1317 |
|
|
set_nz(a); |
1318 |
|
|
c_flag = a & 0x40; |
1319 |
|
|
v_flag = (a & 0x40) ^ ((a & 0x20) << 1); |
1320 |
|
|
} else { |
1321 |
|
|
n_flag = c_flag ? 0x80 : 0; |
1322 |
|
|
z_flag = a; |
1323 |
|
|
v_flag = (tmp2 ^ a) & 0x40; |
1324 |
|
|
if ((tmp2 & 0x0f) + (tmp2 & 0x01) > 5) |
1325 |
|
|
a = a & 0xf0 | (a + 6) & 0x0f; |
1326 |
|
|
if (c_flag = ((tmp2 + (tmp2 & 0x10)) & 0x1f0) > 0x50) |
1327 |
|
|
a += 0x60; |
1328 |
|
|
} |
1329 |
|
|
ENDOP(2); |
1330 |
|
|
|
1331 |
|
|
case 0x8b: // ANE #imm |
1332 |
|
|
set_nz(a = read_byte_imm() & x & (a | 0xee)); |
1333 |
|
|
ENDOP(2); |
1334 |
|
|
|
1335 |
|
|
case 0x93: // SHA (ind),Y |
1336 |
|
|
#if PC_IS_POINTER |
1337 |
|
|
tmp2 = read_zp(pc[0] + 1); |
1338 |
|
|
#else |
1339 |
|
|
tmp2 = read_zp(read_byte(pc) + 1); |
1340 |
|
|
#endif |
1341 |
|
|
write_byte(read_adr_ind_y(), a & x & (tmp2+1)); |
1342 |
|
|
ENDOP(6); |
1343 |
|
|
|
1344 |
|
|
case 0x9b: // SHS abs,Y |
1345 |
|
|
#if PC_IS_POINTER |
1346 |
|
|
tmp2 = pc[1]; |
1347 |
|
|
#else |
1348 |
|
|
tmp2 = read_byte(pc+1); |
1349 |
|
|
#endif |
1350 |
|
|
write_byte(read_adr_abs_y(), a & x & (tmp2+1)); |
1351 |
|
|
sp = a & x; |
1352 |
|
|
ENDOP(5); |
1353 |
|
|
|
1354 |
|
|
case 0x9c: // SHY abs,X |
1355 |
|
|
#if PC_IS_POINTER |
1356 |
|
|
tmp2 = pc[1]; |
1357 |
|
|
#else |
1358 |
|
|
tmp2 = read_byte(pc+1); |
1359 |
|
|
#endif |
1360 |
|
|
write_byte(read_adr_abs_x(), y & (tmp2+1)); |
1361 |
|
|
ENDOP(5); |
1362 |
|
|
|
1363 |
|
|
case 0x9e: // SHX abs,Y |
1364 |
|
|
#if PC_IS_POINTER |
1365 |
|
|
tmp2 = pc[1]; |
1366 |
|
|
#else |
1367 |
|
|
tmp2 = read_byte(pc+1); |
1368 |
|
|
#endif |
1369 |
|
|
write_byte(read_adr_abs_y(), x & (tmp2+1)); |
1370 |
|
|
ENDOP(5); |
1371 |
|
|
|
1372 |
|
|
case 0x9f: // SHA abs,Y |
1373 |
|
|
#if PC_IS_POINTER |
1374 |
|
|
tmp2 = pc[1]; |
1375 |
|
|
#else |
1376 |
|
|
tmp2 = read_byte(pc+1); |
1377 |
|
|
#endif |
1378 |
|
|
write_byte(read_adr_abs_y(), a & x & (tmp2+1)); |
1379 |
|
|
ENDOP(5); |
1380 |
|
|
|
1381 |
|
|
case 0xab: // LXA #imm |
1382 |
|
|
set_nz(a = x = (a | 0xee) & read_byte_imm()); |
1383 |
|
|
ENDOP(2); |
1384 |
|
|
|
1385 |
|
|
case 0xbb: // LAS abs,Y |
1386 |
|
|
set_nz(a = x = sp = read_byte_abs_y() & sp); |
1387 |
|
|
ENDOP(4); |
1388 |
|
|
|
1389 |
|
|
case 0xcb: // SBX #imm |
1390 |
|
|
x &= a; |
1391 |
|
|
adr = x - read_byte_imm(); |
1392 |
|
|
c_flag = adr < 0x100; |
1393 |
|
|
set_nz(x = adr); |
1394 |
|
|
ENDOP(2); |
1395 |
|
|
|
1396 |
|
|
case 0x02: |
1397 |
|
|
case 0x12: |
1398 |
|
|
case 0x22: |
1399 |
|
|
case 0x32: |
1400 |
|
|
case 0x42: |
1401 |
|
|
case 0x52: |
1402 |
|
|
case 0x62: |
1403 |
|
|
case 0x72: |
1404 |
|
|
case 0x92: |
1405 |
|
|
case 0xb2: |
1406 |
|
|
case 0xd2: |
1407 |
|
|
#if PC_IS_POINTER |
1408 |
|
|
illegal_op(*(pc-1), pc-pc_base-1); |
1409 |
|
|
#else |
1410 |
|
|
illegal_op(read_byte(pc-1), pc-1); |
1411 |
|
|
#endif |
1412 |
|
|
break; |