mirror of
https://github.com/holub/mame
synced 2025-06-24 05:17:04 +03:00

- Remove irq_line methods from M6502, M6800, M6809, S2600 and replace uses with DEVCB_INPUTLINE - Remove a few IRQ passthroughs from spiders.cpp - Add several aliases for M6800_IRQ_LINE
333 lines
11 KiB
C++
333 lines
11 KiB
C++
// license:BSD-3-Clause
|
|
// copyright-holders:Olivier Galibert
|
|
/***************************************************************************
|
|
|
|
m6502.h
|
|
|
|
Mostek 6502, original NMOS variant
|
|
|
|
***************************************************************************/
|
|
|
|
#ifndef __M6502FAM_H__
|
|
#define __M6502FAM_H__
|
|
|
|
#define MCFG_M6502_DISABLE_DIRECT() \
|
|
downcast<m6502_device *>(device)->disable_direct();
|
|
|
|
#define MCFG_M6502_SYNC_CALLBACK(_cb) \
|
|
devcb = &m6502_device::set_sync_callback(*device, DEVCB_##_cb);
|
|
|
|
class m6502_device : public cpu_device {
|
|
public:
|
|
enum {
|
|
IRQ_LINE = INPUT_LINE_IRQ0,
|
|
APU_IRQ_LINE = INPUT_LINE_IRQ1,
|
|
NMI_LINE = INPUT_LINE_NMI,
|
|
V_LINE = INPUT_LINE_IRQ0 + 16
|
|
};
|
|
|
|
m6502_device(const machine_config &mconfig, const char *tag, device_t *owner, UINT32 clock);
|
|
m6502_device(const machine_config &mconfig, device_type type, const char *name, const char *tag, device_t *owner, UINT32 clock, const char *shortname, const char *source);
|
|
|
|
bool get_sync() const { return sync; }
|
|
void disable_direct() { direct_disabled = true; }
|
|
|
|
template<class _Object> static devcb_base &set_sync_callback(device_t &device, _Object object) { return downcast<m6502_device &>(device).sync_w.set_callback(object); }
|
|
|
|
devcb_write_line sync_w;
|
|
|
|
protected:
|
|
class memory_interface {
|
|
public:
|
|
address_space *program, *sprogram;
|
|
direct_read_data *direct, *sdirect;
|
|
|
|
virtual ~memory_interface() {}
|
|
virtual UINT8 read(UINT16 adr) = 0;
|
|
virtual UINT8 read_9(UINT16 adr);
|
|
virtual UINT8 read_sync(UINT16 adr) = 0;
|
|
virtual UINT8 read_arg(UINT16 adr) = 0;
|
|
virtual void write(UINT16 adr, UINT8 val) = 0;
|
|
virtual void write_9(UINT16 adr, UINT8 val);
|
|
};
|
|
|
|
class mi_default_normal : public memory_interface {
|
|
public:
|
|
virtual ~mi_default_normal() {}
|
|
virtual UINT8 read(UINT16 adr) override;
|
|
virtual UINT8 read_sync(UINT16 adr) override;
|
|
virtual UINT8 read_arg(UINT16 adr) override;
|
|
virtual void write(UINT16 adr, UINT8 val) override;
|
|
};
|
|
|
|
class mi_default_nd : public mi_default_normal {
|
|
public:
|
|
virtual ~mi_default_nd() {}
|
|
virtual UINT8 read_sync(UINT16 adr) override;
|
|
virtual UINT8 read_arg(UINT16 adr) override;
|
|
};
|
|
|
|
struct disasm_entry {
|
|
const char *opcode;
|
|
int mode;
|
|
offs_t flags;
|
|
};
|
|
|
|
enum {
|
|
STATE_RESET = 0xff00
|
|
};
|
|
|
|
enum {
|
|
DASM_non, /* no additional arguments */
|
|
DASM_aba, /* absolute */
|
|
DASM_abx, /* absolute + X */
|
|
DASM_aby, /* absolute + Y */
|
|
DASM_acc, /* accumulator */
|
|
DASM_adr, /* absolute address (jmp,jsr) */
|
|
DASM_bzp, /* zero page with bit selection */
|
|
DASM_iax, /* indirect + X (65c02 jmp) */
|
|
DASM_idx, /* zero page pre indexed */
|
|
DASM_idy, /* zero page post indexed */
|
|
DASM_idz, /* zero page post indexed (65ce02) */
|
|
DASM_imm, /* immediate */
|
|
DASM_imp, /* implicit */
|
|
DASM_ind, /* indirect (jmp) */
|
|
DASM_isy, /* zero page pre indexed sp and post indexed Y (65ce02) */
|
|
DASM_iw2, /* immediate word (65ce02) */
|
|
DASM_iw3, /* augment (65ce02) */
|
|
DASM_rel, /* relative */
|
|
DASM_rw2, /* relative word (65cs02, 65ce02) */
|
|
DASM_zpb, /* zero page and branch (65c02 bbr, bbs) */
|
|
DASM_zpg, /* zero page */
|
|
DASM_zpi, /* zero page indirect (65c02) */
|
|
DASM_zpx, /* zero page + X */
|
|
DASM_zpy, /* zero page + Y */
|
|
DASM_imz, /* load immediate byte, store to zero page address (M740) */
|
|
DASM_spg, /* "special page": implied FF00 OR immediate value (M740)*/
|
|
DASM_biz, /* bit, zero page (M740) */
|
|
DASM_bzr, /* bit, zero page, relative offset (M740) */
|
|
DASM_bar, /* bit, accumulator, relative offset (M740) */
|
|
DASM_bac /* bit, accumulator (M740) */
|
|
};
|
|
|
|
enum {
|
|
F_N = 0x80,
|
|
F_V = 0x40,
|
|
F_E = 0x20, // 65ce02
|
|
F_T = 0x20, // M740: replaces A with $00,X in some opcodes when set
|
|
F_B = 0x10,
|
|
F_D = 0x08,
|
|
F_I = 0x04,
|
|
F_Z = 0x02,
|
|
F_C = 0x01
|
|
};
|
|
|
|
virtual void init();
|
|
|
|
// device-level overrides
|
|
virtual void device_start() override;
|
|
virtual void device_reset() override;
|
|
|
|
// device_execute_interface overrides
|
|
virtual UINT32 execute_min_cycles() const override;
|
|
virtual UINT32 execute_max_cycles() const override;
|
|
virtual UINT32 execute_input_lines() const override;
|
|
virtual void execute_run() override;
|
|
virtual void execute_set_input(int inputnum, int state) override;
|
|
|
|
// device_memory_interface overrides
|
|
virtual const address_space_config *memory_space_config(address_spacenum spacenum = AS_0) const override;
|
|
|
|
// device_state_interface overrides
|
|
virtual void state_import(const device_state_entry &entry) override;
|
|
virtual void state_export(const device_state_entry &entry) override;
|
|
virtual void state_string_export(const device_state_entry &entry, std::string &str) const override;
|
|
|
|
// device_disasm_interface overrides
|
|
virtual UINT32 disasm_min_opcode_bytes() const override;
|
|
virtual UINT32 disasm_max_opcode_bytes() const override;
|
|
virtual offs_t disasm_disassemble(char *buffer, offs_t pc, const UINT8 *oprom, const UINT8 *opram, UINT32 options) override;
|
|
|
|
address_space_config program_config, sprogram_config;
|
|
|
|
UINT16 PPC; /* previous program counter */
|
|
UINT16 NPC; /* next start-of-instruction program counter */
|
|
UINT16 PC; /* program counter */
|
|
UINT16 SP; /* stack pointer (always 100 - 1FF) */
|
|
UINT16 TMP; /* temporary internal values */
|
|
UINT8 TMP2; /* another temporary internal value, 8 bits this time */
|
|
UINT8 A; /* Accumulator */
|
|
UINT8 X; /* X index register */
|
|
UINT8 Y; /* Y index register */
|
|
UINT8 P; /* Processor status */
|
|
UINT8 IR; /* Prefetched instruction register */
|
|
int inst_state_base; /* Current instruction bank */
|
|
|
|
memory_interface *mintf;
|
|
int inst_state, inst_substate;
|
|
int icount;
|
|
bool nmi_state, irq_state, apu_irq_state, v_state;
|
|
bool irq_taken, sync, direct_disabled, inhibit_interrupts;
|
|
|
|
static const disasm_entry disasm_entries[0x100];
|
|
|
|
offs_t disassemble_generic(char *buffer, offs_t pc, const UINT8 *oprom, const UINT8 *opram, UINT32 options, const disasm_entry *table);
|
|
UINT8 read(UINT16 adr) { return mintf->read(adr); }
|
|
UINT8 read_9(UINT16 adr) { return mintf->read_9(adr); }
|
|
void write(UINT16 adr, UINT8 val) { mintf->write(adr, val); }
|
|
void write_9(UINT16 adr, UINT8 val) { mintf->write_9(adr, val); }
|
|
UINT8 read_arg(UINT16 adr) { return mintf->read_arg(adr); }
|
|
UINT8 read_pc() { return mintf->read_arg(PC++); }
|
|
UINT8 read_pc_noinc() { return mintf->read_arg(PC); }
|
|
void prefetch();
|
|
void prefetch_noirq();
|
|
void set_nz(UINT8 v);
|
|
|
|
virtual void do_exec_full();
|
|
virtual void do_exec_partial();
|
|
|
|
// inline helpers
|
|
static inline bool page_changing(UINT16 base, int delta) { return ((base + delta) ^ base) & 0xff00; }
|
|
static inline UINT16 set_l(UINT16 base, UINT8 val) { return (base & 0xff00) | val; }
|
|
static inline UINT16 set_h(UINT16 base, UINT8 val) { return (base & 0x00ff) | (val << 8); }
|
|
|
|
inline void dec_SP() { SP = set_l(SP, SP-1); }
|
|
inline void inc_SP() { SP = set_l(SP, SP+1); }
|
|
|
|
void do_adc_d(UINT8 val);
|
|
void do_adc_nd(UINT8 val);
|
|
void do_sbc_d(UINT8 val);
|
|
void do_sbc_nd(UINT8 val);
|
|
void do_arr_d();
|
|
void do_arr_nd();
|
|
|
|
void do_adc(UINT8 val);
|
|
void do_cmp(UINT8 val1, UINT8 val2);
|
|
void do_sbc(UINT8 val);
|
|
void do_bit(UINT8 val);
|
|
void do_arr();
|
|
UINT8 do_asl(UINT8 v);
|
|
UINT8 do_lsr(UINT8 v);
|
|
UINT8 do_ror(UINT8 v);
|
|
UINT8 do_rol(UINT8 v);
|
|
UINT8 do_asr(UINT8 v);
|
|
|
|
#define O(o) void o ## _full(); void o ## _partial()
|
|
|
|
// NMOS 6502 opcodes
|
|
// documented opcodes
|
|
O(adc_aba); O(adc_abx); O(adc_aby); O(adc_idx); O(adc_idy); O(adc_imm); O(adc_zpg); O(adc_zpx);
|
|
O(and_aba); O(and_abx); O(and_aby); O(and_imm); O(and_idx); O(and_idy); O(and_zpg); O(and_zpx);
|
|
O(asl_aba); O(asl_abx); O(asl_acc); O(asl_zpg); O(asl_zpx);
|
|
O(bcc_rel);
|
|
O(bcs_rel);
|
|
O(beq_rel);
|
|
O(bit_aba); O(bit_zpg);
|
|
O(bmi_rel);
|
|
O(bne_rel);
|
|
O(bpl_rel);
|
|
O(brk_imp);
|
|
O(bvc_rel);
|
|
O(bvs_rel);
|
|
O(clc_imp);
|
|
O(cld_imp);
|
|
O(cli_imp);
|
|
O(clv_imp);
|
|
O(cmp_aba); O(cmp_abx); O(cmp_aby); O(cmp_idx); O(cmp_idy); O(cmp_imm); O(cmp_zpg); O(cmp_zpx);
|
|
O(cpx_aba); O(cpx_imm); O(cpx_zpg);
|
|
O(cpy_aba); O(cpy_imm); O(cpy_zpg);
|
|
O(dec_aba); O(dec_abx); O(dec_zpg); O(dec_zpx);
|
|
O(dex_imp);
|
|
O(dey_imp);
|
|
O(eor_aba); O(eor_abx); O(eor_aby); O(eor_idx); O(eor_idy); O(eor_imm); O(eor_zpg); O(eor_zpx);
|
|
O(inc_aba); O(inc_abx); O(inc_zpg); O(inc_zpx);
|
|
O(inx_imp);
|
|
O(iny_imp);
|
|
O(jmp_adr); O(jmp_ind);
|
|
O(jsr_adr);
|
|
O(lda_aba); O(lda_abx); O(lda_aby); O(lda_idx); O(lda_idy); O(lda_imm); O(lda_zpg); O(lda_zpx);
|
|
O(ldx_aba); O(ldx_aby); O(ldx_imm); O(ldx_zpg); O(ldx_zpy);
|
|
O(ldy_aba); O(ldy_abx); O(ldy_imm); O(ldy_zpg); O(ldy_zpx);
|
|
O(lsr_aba); O(lsr_abx); O(lsr_acc); O(lsr_zpg); O(lsr_zpx);
|
|
O(nop_imp);
|
|
O(ora_aba); O(ora_abx); O(ora_aby); O(ora_imm); O(ora_idx); O(ora_idy); O(ora_zpg); O(ora_zpx);
|
|
O(pha_imp);
|
|
O(php_imp);
|
|
O(pla_imp);
|
|
O(plp_imp);
|
|
O(rol_aba); O(rol_abx); O(rol_acc); O(rol_zpg); O(rol_zpx);
|
|
O(ror_aba); O(ror_abx); O(ror_acc); O(ror_zpg); O(ror_zpx);
|
|
O(rti_imp);
|
|
O(rts_imp);
|
|
O(sbc_aba); O(sbc_abx); O(sbc_aby); O(sbc_idx); O(sbc_idy); O(sbc_imm); O(sbc_zpg); O(sbc_zpx);
|
|
O(sec_imp);
|
|
O(sed_imp);
|
|
O(sei_imp);
|
|
O(sta_aba); O(sta_abx); O(sta_aby); O(sta_idx); O(sta_idy); O(sta_zpg); O(sta_zpx);
|
|
O(stx_aba); O(stx_zpg); O(stx_zpy);
|
|
O(sty_aba); O(sty_zpg); O(sty_zpx);
|
|
O(tax_imp);
|
|
O(tay_imp);
|
|
O(tsx_imp);
|
|
O(txa_imp);
|
|
O(txs_imp);
|
|
O(tya_imp);
|
|
|
|
// exceptions
|
|
O(reset);
|
|
|
|
// undocumented reliable instructions
|
|
O(dcp_aba); O(dcp_abx); O(dcp_aby); O(dcp_idx); O(dcp_idy); O(dcp_zpg); O(dcp_zpx);
|
|
O(isb_aba); O(isb_abx); O(isb_aby); O(isb_idx); O(isb_idy); O(isb_zpg); O(isb_zpx);
|
|
O(lax_aba); O(lax_aby); O(lax_idx); O(lax_idy); O(lax_zpg); O(lax_zpy);
|
|
O(rla_aba); O(rla_abx); O(rla_aby); O(rla_idx); O(rla_idy); O(rla_zpg); O(rla_zpx);
|
|
O(rra_aba); O(rra_abx); O(rra_aby); O(rra_idx); O(rra_idy); O(rra_zpg); O(rra_zpx);
|
|
O(sax_aba); O(sax_idx); O(sax_zpg); O(sax_zpy);
|
|
O(sbx_imm);
|
|
O(sha_aby); O(sha_idy);
|
|
O(shs_aby);
|
|
O(shx_aby);
|
|
O(shy_abx);
|
|
O(slo_aba); O(slo_abx); O(slo_aby); O(slo_idx); O(slo_idy); O(slo_zpg); O(slo_zpx);
|
|
O(sre_aba); O(sre_abx); O(sre_aby); O(sre_idx); O(sre_idy); O(sre_zpg); O(sre_zpx);
|
|
|
|
// undocumented unreliable instructions
|
|
// behaviour differs between visual6502 and online docs, which
|
|
// is a clear sign reliability is not to be expected
|
|
// implemented version follows visual6502
|
|
O(anc_imm);
|
|
O(ane_imm);
|
|
O(arr_imm);
|
|
O(asr_imm);
|
|
O(las_aby);
|
|
O(lxa_imm);
|
|
|
|
// nop variants
|
|
O(nop_imm); O(nop_aba); O(nop_abx); O(nop_zpg); O(nop_zpx);
|
|
|
|
// system killers
|
|
O(kil_non);
|
|
|
|
#undef O
|
|
};
|
|
|
|
enum {
|
|
M6502_PC = 1,
|
|
M6502_A,
|
|
M6502_X,
|
|
M6502_Y,
|
|
M6502_P,
|
|
M6502_S,
|
|
M6502_IR
|
|
};
|
|
|
|
enum {
|
|
M6502_IRQ_LINE = m6502_device::IRQ_LINE,
|
|
M6502_NMI_LINE = m6502_device::NMI_LINE,
|
|
M6502_SET_OVERFLOW = m6502_device::V_LINE
|
|
};
|
|
|
|
extern const device_type M6502;
|
|
|
|
#endif
|