2017-01-16 18:35:28 +00:00
|
|
|
#include <stdio.h>
|
|
|
|
#include <stdint.h>
|
|
|
|
#include <stdlib.h>
|
|
|
|
#include <stdbool.h>
|
|
|
|
|
|
|
|
#include <assert.h>
|
|
|
|
#include <string.h>
|
|
|
|
|
|
|
|
#include "bus.h"
|
2017-01-17 13:29:20 +00:00
|
|
|
#include "instruction.h"
|
|
|
|
#include "rk65c02.h"
|
2017-01-16 18:35:28 +00:00
|
|
|
|
2017-01-26 12:11:00 +00:00
|
|
|
void rk65c02_exec(rk65c02emu_t *);
|
|
|
|
|
2017-01-27 09:13:32 +00:00
|
|
|
/*
|
|
|
|
* Prepare the emulator for use, set initial CPU state.
|
|
|
|
*/
|
2017-01-18 16:18:19 +00:00
|
|
|
rk65c02emu_t
|
|
|
|
rk65c02_init(bus_t *b)
|
|
|
|
{
|
|
|
|
rk65c02emu_t e;
|
|
|
|
|
|
|
|
e.bus = b;
|
|
|
|
e.state = STOPPED;
|
2017-01-25 20:17:18 +00:00
|
|
|
e.regs.P = P_UNDEFINED|P_IRQ_DISABLE;
|
2017-01-18 16:18:19 +00:00
|
|
|
|
|
|
|
return e;
|
|
|
|
}
|
2017-01-16 18:35:28 +00:00
|
|
|
|
2017-01-27 09:13:32 +00:00
|
|
|
/*
|
|
|
|
* Execute a single instruction within emulator.
|
|
|
|
*/
|
2017-01-16 18:35:28 +00:00
|
|
|
void
|
2017-01-26 12:11:00 +00:00
|
|
|
rk65c02_exec(rk65c02emu_t *e)
|
|
|
|
{
|
2017-01-16 18:35:28 +00:00
|
|
|
instruction_t i;
|
2017-01-18 16:18:19 +00:00
|
|
|
instrdef_t id;
|
2017-01-16 18:35:28 +00:00
|
|
|
|
2017-01-26 12:11:00 +00:00
|
|
|
/* XXX: handle breakpoints and watch points */
|
|
|
|
|
|
|
|
/* if disassembly-when-running enabled */
|
|
|
|
disassemble(e->bus, e->regs.PC);
|
|
|
|
|
|
|
|
i = instruction_fetch(e->bus, e->regs.PC);
|
|
|
|
id = instruction_decode(i.opcode);
|
|
|
|
|
|
|
|
if (id.emul != NULL) {
|
|
|
|
id.emul(e, &id, &i);
|
2017-01-27 19:43:08 +00:00
|
|
|
if (!instruction_modify_pc(&id))
|
|
|
|
program_counter_increment(e, &id);
|
2017-01-26 12:11:00 +00:00
|
|
|
} else {
|
|
|
|
printf("unimplemented opcode %X @ %X\n", i.opcode,
|
|
|
|
e->regs.PC);
|
|
|
|
e->state = STOPPED;
|
|
|
|
e->stopreason = EMUERROR;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-01-27 09:13:32 +00:00
|
|
|
/*
|
|
|
|
* Start the emulator.
|
|
|
|
*/
|
2017-01-26 12:11:00 +00:00
|
|
|
void
|
|
|
|
rk65c02_start(rk65c02emu_t *e) {
|
|
|
|
|
2017-01-18 16:18:19 +00:00
|
|
|
e->state = RUNNING;
|
|
|
|
while (e->state == RUNNING) {
|
2017-01-26 12:11:00 +00:00
|
|
|
rk65c02_exec(e);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-01-27 09:13:32 +00:00
|
|
|
/*
|
|
|
|
* Execute as many instructions as specified in steps argument.
|
|
|
|
*/
|
2017-01-26 12:11:00 +00:00
|
|
|
void
|
|
|
|
rk65c02_step(rk65c02emu_t *e, uint16_t steps) {
|
|
|
|
|
|
|
|
uint16_t i = 0;
|
|
|
|
|
|
|
|
e->state = STEPPING;
|
|
|
|
while ((e->state == STEPPING) && (i < steps)) {
|
|
|
|
rk65c02_exec(e);
|
|
|
|
i++;
|
2017-01-16 18:35:28 +00:00
|
|
|
}
|
|
|
|
}
|
2017-01-27 10:27:14 +00:00
|
|
|
|
|
|
|
void
|
|
|
|
rk65c02_dump_regs(rk65c02emu_t *e)
|
|
|
|
{
|
2017-01-27 21:28:09 +00:00
|
|
|
printf("A: %X X: %X Y: %X PC: %X SP: %X P: ",
|
|
|
|
e->regs.A, e->regs.X, e->regs.Y, e->regs.PC, e->regs.SP);
|
2017-01-27 10:27:14 +00:00
|
|
|
|
|
|
|
if (e->regs.P & P_NEGATIVE)
|
|
|
|
printf("N");
|
|
|
|
else
|
|
|
|
printf("-");
|
|
|
|
|
|
|
|
if (e->regs.P & P_SIGN_OVERFLOW)
|
|
|
|
printf("O");
|
|
|
|
else
|
|
|
|
printf("-");
|
|
|
|
|
|
|
|
if (e->regs.P & P_UNDEFINED)
|
|
|
|
printf("1");
|
|
|
|
else
|
|
|
|
printf("-");
|
|
|
|
|
|
|
|
if (e->regs.P & P_BREAK)
|
|
|
|
printf("B");
|
|
|
|
else
|
|
|
|
printf("-");
|
|
|
|
|
|
|
|
if (e->regs.P & P_DECIMAL)
|
|
|
|
printf("D");
|
|
|
|
else
|
|
|
|
printf("-");
|
|
|
|
|
|
|
|
if (e->regs.P & P_IRQ_DISABLE)
|
|
|
|
printf("I");
|
|
|
|
else
|
|
|
|
printf("-");
|
|
|
|
|
|
|
|
if (e->regs.P & P_ZERO)
|
|
|
|
printf("Z");
|
|
|
|
else
|
|
|
|
printf("-");
|
|
|
|
|
|
|
|
if (e->regs.P & P_CARRY)
|
|
|
|
printf("C");
|
|
|
|
else
|
|
|
|
printf("-");
|
|
|
|
|
|
|
|
printf("\n");
|
|
|
|
}
|
2017-01-16 22:54:24 +00:00
|
|
|
/*
|
2017-01-16 18:35:28 +00:00
|
|
|
int
|
|
|
|
main(void)
|
|
|
|
{
|
|
|
|
bus_t b;
|
|
|
|
|
|
|
|
b = bus_init();
|
|
|
|
|
|
|
|
bus_write_1(&b, 0, OP_INX);
|
|
|
|
bus_write_1(&b, 1, OP_NOP);
|
|
|
|
bus_write_1(&b, 2, OP_LDY_IMM);
|
|
|
|
bus_write_1(&b, 3, 0x1);
|
|
|
|
bus_write_1(&b, 4, OP_TSB_ZP);
|
|
|
|
bus_write_1(&b, 5, 0x3);
|
|
|
|
bus_write_1(&b, 6, OP_JSR);
|
|
|
|
bus_write_1(&b, 7, 0x09);
|
|
|
|
bus_write_1(&b, 8, 0x0);
|
|
|
|
bus_write_1(&b, 9, OP_STP);
|
|
|
|
|
|
|
|
rk6502_start(&b, 0);
|
|
|
|
|
|
|
|
bus_finish(&b);
|
|
|
|
}
|
2017-01-16 22:54:24 +00:00
|
|
|
*/
|