mame/src/emu/cpu/dsp56k/dsp56k.c
Aaron Giles e7c418ef0f Generalized the concept of opbase access into "direct" access.
Removed opbase globals to the address_space structure.
Cleaned up names of pointers (decrypted and raw versus rom and ram).
Added inline functions to read/write data via any address space.
Added macros for existing functions to point them to the new functions.
Other related cleanups.
2008-11-17 18:33:10 +00:00

774 lines
27 KiB
C

/***************************************************************************
dsp56k.c
Core implementation for the portable DSP56k emulator.
Written by Andrew Gardner
****************************************************************************
Note:
This CPU emulator is very much a work-in-progress.
DONE:
1: 1, 2, 3, 4, 5, 6, 7, 8, 9, 10,
11, , , , , , ,18, , ,
TODO:
X 1-6 Explore CORE naming scheme.
- 1-9 paragraph 1 : memory access timings
- 1-9 Data ALU arithmetic operations generally use fractional two's complement arithmetic
(Unsigned numbers are only supported by the multiply and multiply-accumulate instruction)
- 1-9 For fractional arithmetic, the 31-bit product is added to the 40-bit contents of A or B. No pipeline!
- 1-10 Two types of rounding: convergent rounding and two's complement rounding. See status register bit R.
- 1-10 Logic unit is 16-bits wide and works on MSP portion of accum register
- 1-10 The AGU can implement three types of arithmetic: linear, modulo, and reverse carry.
- 1-12 "Two external interrupt pins!!!"
- 1-12 Take care of all interrupt priority (IPR) stuff!
- 1-19 Memory WAIT states
- 1-20 The timer's interesting!
- 1-21 Vectored exception requests on the Host Interface!
***************************************************************************/
#include "debugger.h"
#include "deprecat.h"
#include "dsp56k.h"
#include "dsp56def.h"
/***************************************************************************
LOCAL DECLARATIONS
***************************************************************************/
static CPU_INIT( dsp56k );
static CPU_RESET( dsp56k );
static CPU_EXIT( dsp56k );
/***************************************************************************
MACROS
***************************************************************************/
/***************************************************************************
STRUCTURES & TYPEDEFS
***************************************************************************/
// 5-4 Host Interface
typedef struct
{
// **** Dsp56k side **** //
// Host Control Register
UINT16* hcr;
// Host Status Register
UINT16* hsr;
// Host Transmit/Receive Data
UINT16* htrx;
// **** Host CPU side **** //
// Interrupt Control Register
UINT8 icr;
// Command Vector Register
UINT8 cvr;
// Interrupt Status Register
UINT8 isr;
// Interrupt Vector Register
UINT8 ivr;
// Transmit / Receive Registers
UINT8 trxh;
UINT8 trxl;
// HACK - Host interface bootstrap write offset
UINT16 bootstrap_offset;
} dsp56k_host_interface;
// 1-9 ALU
typedef struct
{
// Four 16-bit input registers (can be accessed as 2 32-bit registers)
PAIR x;
PAIR y;
// Two 32-bit accumulator registers + 8-bit accumulator extension registers
PAIR64 a;
PAIR64 b;
// An accumulation shifter
// One data bus shifter/limiter
// A parallel, single cycle, non-pipelined Multiply-Accumulator (MAC) unit
// Basics
} dsp56k_data_alu;
// 1-10 Address Generation Unit (AGU)
typedef struct
{
// Four address registers
UINT16 r0;
UINT16 r1;
UINT16 r2;
UINT16 r3;
// Four offset registers
UINT16 n0;
UINT16 n1;
UINT16 n2;
UINT16 n3;
// Four modifier registers
UINT16 m0;
UINT16 m1;
UINT16 m2;
UINT16 m3;
// Used in loop processing
UINT16 temp;
// FM.4-5 - hmmm?
// UINT8 status;
// Basics
} dsp56k_agu;
// 1-11 Program Control Unit (PCU)
typedef struct
{
// Program Counter
UINT16 pc;
// Loop Address
UINT16 la;
// Loop Counter
UINT16 lc;
// Status Register
UINT16 sr;
// Operating Mode Register
UINT16 omr;
// Stack Pointer
UINT16 sp;
// Stack (TODO: 15-level?)
PAIR ss[16];
// Controls IRQ processing
void (*service_interrupts)(void);
// A list of pending interrupts (indices into dsp56k_interrupt_sources array)
INT8 pending_interrupts[32];
// Basics
// Other PCU internals
UINT16 reset_vector;
} dsp56k_pcu;
// 1-8 The dsp56156 CORE
typedef struct
{
// PROGRAM CONTROLLER
dsp56k_pcu PCU;
// ADR ALU (AGU)
dsp56k_agu AGU;
// CLOCK GEN
//static emu_timer *dsp56k_timer; // 1-5, 1-8 - Clock gen
// DATA ALU
dsp56k_data_alu ALU;
// OnCE
// IBS and BITFIELD UNIT
// Host Interface
dsp56k_host_interface HI;
// IRQ line states
UINT8 modA_state;
UINT8 modB_state;
UINT8 modC_state;
UINT8 reset_state;
// HACK - Bootstrap mode state variable.
UINT8 bootstrap_mode;
UINT8 repFlag; // Knowing if we're in a 'repeat' state (dunno how the processor does this)
UINT32 repAddr; // The address of the instruction to repeat...
/* MAME internal stuff */
UINT32 ppc;
UINT32 op;
int interrupt_cycles;
void (*output_pins_changed)(UINT32 pins);
const device_config *device;
} dsp56k_core;
/***************************************************************************
ONBOARD MEMORY ALLOCATION
***************************************************************************/
static UINT16 *dsp56k_peripheral_ram;
static UINT16 *dsp56k_program_ram;
/***************************************************************************
FUNCTION PROTOTYPES
***************************************************************************/
/***************************************************************************
PRIVATE GLOBAL VARIABLES
***************************************************************************/
static dsp56k_core core;
static int dsp56k_icount;
/***************************************************************************
COMPONENT FUNCTIONALITY
***************************************************************************/
// 1-9 ALU
// #include "dsp56alu.c"
// 1-10 Address Generation Unit (AGU)
// #include "dsp56agu.c"
// 1-11 Program Control Unit (PCU)
#include "dsp56pcu.c"
// 5-1 Host Interface (HI)
//#include "dsp56hi.c"
// 4-8 Memory handlers for on-chip peripheral memory.
#include "dsp56mem.c"
/***************************************************************************
MEMORY ACCESSORS
***************************************************************************/
#define ROPCODE(pc) program_decrypted_read_word(pc)
/***************************************************************************
EXECEPTION HANDLING
***************************************************************************/
#ifdef UNUSED_FUNCTION
INLINE void generate_exception(int exception)
{
}
#endif
#ifdef UNUSED_FUNCTION
INLINE void invalid_instruction(UINT32 op)
{
}
#endif
/***************************************************************************
IRQ HANDLING
***************************************************************************/
static void check_irqs(void)
{
logerror("Dsp56k check irqs\n");
}
static void set_irq_line(int irqline, int state)
{
logerror("DSP56k set irq line %d %d\n", irqline, state);
/* I seem to recall this is an okay thing to do */
if (state == PULSE_LINE)
{
logerror("WARNING: The dsp56k does not support pulsed interrupts.\n");
return;
}
switch(irqline)
{
case DSP56K_IRQ_MODA:
// TODO: 1-12 Get this triggering right
if (irqa_trigger())
logerror("DSP56k IRQA is set to fire on the \"Negative Edge\".\n");
if (state != CLEAR_LINE)
core.modA_state = TRUE;
else
core.modA_state = FALSE;
if (core.reset_state != TRUE)
dsp56k_add_pending_interrupt("IRQA");
break;
case DSP56K_IRQ_MODB:
// TODO: 1-12 Get this triggering right
if (irqb_trigger())
logerror("DSP56k IRQB is set to fire on the \"Negative Edge\".\n");
if (state != CLEAR_LINE)
core.modB_state = TRUE;
else
core.modB_state = FALSE;
if (core.reset_state != TRUE)
dsp56k_add_pending_interrupt("IRQB");
break;
case DSP56K_IRQ_MODC:
if (state != CLEAR_LINE)
core.modC_state = TRUE;
else
core.modC_state = FALSE;
// TODO : Set bus mode or whatever
break;
case DSP56K_IRQ_RESET:
if (state != CLEAR_LINE)
core.reset_state = TRUE;
else
{
/* If it changes state from asserted to cleared. Call the reset function. */
if (core.reset_state == TRUE)
cpu_reset_dsp56k(core.device);
core.reset_state = FALSE;
}
// dsp56k_add_pending_interrupt("Hardware RESET");
break;
default:
logerror("DSP56k setting some weird irq line : %d", irqline);
break;
}
// If the reset line isn't asserted, service interrupts
// TODO: Is it right to immediately service interrupts?
//if (core.reset_state != TRUE)
// pcu_service_interrupts();
}
/***************************************************************************
CONTEXT SWITCHING
***************************************************************************/
static CPU_GET_CONTEXT( dsp56k )
{
if (dst)
*(dsp56k_core *)dst = core;
}
static CPU_SET_CONTEXT( dsp56k )
{
if (src)
core = *(dsp56k_core *)src;
check_irqs();
}
/***************************************************************************
INITIALIZATION AND SHUTDOWN
***************************************************************************/
static CPU_INIT( dsp56k )
{
// Call specific module inits
pcu_init(index);
// agu_init(index);
// alu_init(index);
// HACK - You're not in bootstrap mode upon bootup
core.bootstrap_mode = BOOTSTRAP_OFF;
// Clear the irq states
core.modA_state = FALSE;
core.modB_state = FALSE;
core.modC_state = FALSE;
core.reset_state = FALSE;
/* Save the core's state */
// state_save_register_item("dsp56k", device->tag, 0, modA_state);
// ...
//core.config = device->static_config;
//core.irq_callback = irqcallback;
core.device = device;
}
static void agu_reset(void)
{
// FM.4-3
R0 = 0x0000;
R1 = 0x0000;
R2 = 0x0000;
R3 = 0x0000;
N0 = 0x0000;
N1 = 0x0000;
N2 = 0x0000;
N3 = 0x0000;
M0 = 0xffff;
M1 = 0xffff;
M2 = 0xffff;
M3 = 0xffff;
TEMP = 0x0000;
}
static void alu_reset(void)
{
X = 0x00000000;
Y = 0x00000000;
A = 0x0000000000;
B = 0x0000000000;
}
static CPU_RESET( dsp56k )
{
logerror("Dsp56k reset\n");
core.interrupt_cycles = 0;
core.ppc = 0x0000;
core.repFlag = 0;
core.repAddr = 0x0000;
pcu_reset();
mem_reset();
agu_reset();
alu_reset();
/* HACK - Put a jump to 0x0000 at 0x0000 - this keeps the CPU put in MAME */
program_write_word_16le(0x0000, 0x0124);
}
static CPU_EXIT( dsp56k )
{
}
/***************************************************************************
CORE INCLUDE
***************************************************************************/
#define OP (core.op)
#include "dsp56ops.c"
/***************************************************************************
CORE EXECUTION LOOP
***************************************************************************/
static CPU_EXECUTE( dsp56k )
{
/* If reset line is asserted, do nothing */
if (core.reset_state)
return cycles;
// HACK - if you're in bootstrap mode, simply pretend you ate up all your cycles waiting for data.
if (core.bootstrap_mode != BOOTSTRAP_OFF)
return cycles;
dsp56k_icount = cycles;
dsp56k_icount -= core.interrupt_cycles;
core.interrupt_cycles = 0;
while(dsp56k_icount > 0)
{
execute_one();
pcu_service_interrupts(); /* TODO: There is definitely something un-right about this */
}
dsp56k_icount -= core.interrupt_cycles;
core.interrupt_cycles = 0;
return cycles - dsp56k_icount;
}
/***************************************************************************
DISASSEMBLY HOOK
***************************************************************************/
extern CPU_DISASSEMBLE( dsp56k );
/****************************************************************************
* Internal Memory Maps
****************************************************************************/
static ADDRESS_MAP_START( dsp56156_program_map, ADDRESS_SPACE_PROGRAM, 16 )
AM_RANGE(0x0000,0x07ff) AM_RAM AM_BASE(&dsp56k_program_ram) // 1-5
// AM_RANGE(0x2f00,0x2fff) AM_ROM // 1-5 PROM reserved memory. Is this the right spot for it?
ADDRESS_MAP_END
static ADDRESS_MAP_START( dsp56156_x_data_map, ADDRESS_SPACE_DATA, 16 )
AM_RANGE(0x0000,0x07ff) AM_RAM // 1-5
AM_RANGE(0xffc0,0xffff) AM_READWRITE(peripheral_register_r, peripheral_register_w) AM_BASE(&dsp56k_peripheral_ram) // 1-5 On-chip peripheral registers memory mapped in data space
ADDRESS_MAP_END
/**************************************************************************
* Generic set_info/get_info
**************************************************************************/
static CPU_SET_INFO( dsp56k )
{
switch (state)
{
case CPUINFO_INT_INPUT_STATE + DSP56K_IRQ_MODA: set_irq_line(DSP56K_IRQ_MODA, info->i); break;
case CPUINFO_INT_INPUT_STATE + DSP56K_IRQ_MODB: set_irq_line(DSP56K_IRQ_MODB, info->i); break;
case CPUINFO_INT_INPUT_STATE + DSP56K_IRQ_MODC: set_irq_line(DSP56K_IRQ_MODC, info->i); break;
case CPUINFO_INT_INPUT_STATE + DSP56K_IRQ_RESET: set_irq_line(DSP56K_IRQ_RESET, info->i); break;
case CPUINFO_INT_PC:
case CPUINFO_INT_REGISTER + DSP56K_PC: PC = info->i & 0xffff; break;
case CPUINFO_INT_REGISTER + DSP56K_SR: SR = info->i & 0xffff; break;
case CPUINFO_INT_REGISTER + DSP56K_LC: LC = info->i & 0xffff; break;
case CPUINFO_INT_REGISTER + DSP56K_LA: LA = info->i & 0xffff; break;
case CPUINFO_INT_SP:
case CPUINFO_INT_REGISTER + DSP56K_SP: SP = info->i & 0xff; break;
case CPUINFO_INT_REGISTER + DSP56K_OMR: OMR = info->i & 0xff; break;
case CPUINFO_INT_REGISTER + DSP56K_X: X = info->i & 0xffffffff; break;
case CPUINFO_INT_REGISTER + DSP56K_Y: Y = info->i & 0xffffffff; break;
case CPUINFO_INT_REGISTER + DSP56K_A: A = info->i & (UINT64)U64(0xffffffffffffffff); break; // could benefit from a better mask?
case CPUINFO_INT_REGISTER + DSP56K_B: B = info->i & (UINT64)U64(0xffffffffffffffff); break; // could benefit from a better mask?
case CPUINFO_INT_REGISTER + DSP56K_R0: R0 = info->i & 0xffff; break;
case CPUINFO_INT_REGISTER + DSP56K_R1: R1 = info->i & 0xffff; break;
case CPUINFO_INT_REGISTER + DSP56K_R2: R2 = info->i & 0xffff; break;
case CPUINFO_INT_REGISTER + DSP56K_R3: R3 = info->i & 0xffff; break;
case CPUINFO_INT_REGISTER + DSP56K_N0: N0 = info->i & 0xffff; break;
case CPUINFO_INT_REGISTER + DSP56K_N1: N1 = info->i & 0xffff; break;
case CPUINFO_INT_REGISTER + DSP56K_N2: N2 = info->i & 0xffff; break;
case CPUINFO_INT_REGISTER + DSP56K_N3: N3 = info->i & 0xffff; break;
case CPUINFO_INT_REGISTER + DSP56K_M0: M0 = info->i & 0xffff; break;
case CPUINFO_INT_REGISTER + DSP56K_M1: M1 = info->i & 0xffff; break;
case CPUINFO_INT_REGISTER + DSP56K_M2: M2 = info->i & 0xffff; break;
case CPUINFO_INT_REGISTER + DSP56K_M3: M3 = info->i & 0xffff; break;
/* case CPUINFO_INT_REGISTER + DSP56K_TEMP: TEMP = info->i & 0xffff; break; */
/* case CPUINFO_INT_REGISTER + DSP56K_STATUS: STATUS = info->i & 0xff; break; */
/* The CPU stack */
case CPUINFO_INT_REGISTER + DSP56K_ST0: ST0 = info->i & 0xffffffff; break;
case CPUINFO_INT_REGISTER + DSP56K_ST1: ST1 = info->i & 0xffffffff; break;
case CPUINFO_INT_REGISTER + DSP56K_ST2: ST2 = info->i & 0xffffffff; break;
case CPUINFO_INT_REGISTER + DSP56K_ST3: ST3 = info->i & 0xffffffff; break;
case CPUINFO_INT_REGISTER + DSP56K_ST4: ST4 = info->i & 0xffffffff; break;
case CPUINFO_INT_REGISTER + DSP56K_ST5: ST5 = info->i & 0xffffffff; break;
case CPUINFO_INT_REGISTER + DSP56K_ST6: ST6 = info->i & 0xffffffff; break;
case CPUINFO_INT_REGISTER + DSP56K_ST7: ST7 = info->i & 0xffffffff; break;
case CPUINFO_INT_REGISTER + DSP56K_ST8: ST8 = info->i & 0xffffffff; break;
case CPUINFO_INT_REGISTER + DSP56K_ST9: ST9 = info->i & 0xffffffff; break;
case CPUINFO_INT_REGISTER + DSP56K_ST10: ST10 = info->i & 0xffffffff; break;
case CPUINFO_INT_REGISTER + DSP56K_ST11: ST11 = info->i & 0xffffffff; break;
case CPUINFO_INT_REGISTER + DSP56K_ST12: ST12 = info->i & 0xffffffff; break;
case CPUINFO_INT_REGISTER + DSP56K_ST13: ST13 = info->i & 0xffffffff; break;
case CPUINFO_INT_REGISTER + DSP56K_ST14: ST14 = info->i & 0xffffffff; break;
case CPUINFO_INT_REGISTER + DSP56K_ST15: ST15 = info->i & 0xffffffff; break;
}
}
CPU_GET_INFO( dsp56k )
{
switch (state)
{
// --- the following bits of info are returned as 64-bit signed integers ---
case CPUINFO_INT_CONTEXT_SIZE: info->i = sizeof(core); break;
case CPUINFO_INT_INPUT_LINES: info->i = 4; break;
case CPUINFO_INT_DEFAULT_IRQ_VECTOR: info->i = 0x0000; break;
case CPUINFO_INT_ENDIANNESS: info->i = CPU_IS_LE; break;
case CPUINFO_INT_CLOCK_MULTIPLIER: info->i = 1; break;
case CPUINFO_INT_CLOCK_DIVIDER: info->i = 2; break;
case CPUINFO_INT_MIN_INSTRUCTION_BYTES: info->i = 2; break;
case CPUINFO_INT_MAX_INSTRUCTION_BYTES: info->i = 4; break;
case CPUINFO_INT_MIN_CYCLES: info->i = 1; break; // ?
case CPUINFO_INT_MAX_CYCLES: info->i = 8; break; // ?
case CPUINFO_INT_DATABUS_WIDTH + ADDRESS_SPACE_PROGRAM: info->i = 16; break; // 1-5
case CPUINFO_INT_ADDRBUS_WIDTH + ADDRESS_SPACE_PROGRAM: info->i = 16; break; // 1-5
case CPUINFO_INT_ADDRBUS_SHIFT + ADDRESS_SPACE_PROGRAM: info->i = -1; break;
case CPUINFO_INT_DATABUS_WIDTH + ADDRESS_SPACE_DATA: info->i = 16; break;
case CPUINFO_INT_ADDRBUS_WIDTH + ADDRESS_SPACE_DATA: info->i = 16; break;
case CPUINFO_INT_ADDRBUS_SHIFT + ADDRESS_SPACE_DATA: info->i = -1; break;
case CPUINFO_INT_DATABUS_WIDTH + ADDRESS_SPACE_IO: info->i = 0; break;
case CPUINFO_INT_ADDRBUS_WIDTH + ADDRESS_SPACE_IO: info->i = 0; break;
case CPUINFO_INT_ADDRBUS_SHIFT + ADDRESS_SPACE_IO: info->i = 0; break;
case CPUINFO_INT_INPUT_STATE + DSP56K_IRQ_MODA: info->i = DSP56K_IRQ_MODA; break;
case CPUINFO_INT_INPUT_STATE + DSP56K_IRQ_MODB: info->i = DSP56K_IRQ_MODB; break;
case CPUINFO_INT_INPUT_STATE + DSP56K_IRQ_MODC: info->i = DSP56K_IRQ_MODC; break;
case CPUINFO_INT_PC:
case CPUINFO_INT_REGISTER + DSP56K_PC: info->i = PC; break;
case CPUINFO_INT_PREVIOUSPC: info->i = core.ppc; break;
case CPUINFO_INT_REGISTER + DSP56K_SR: info->i = SR; break;
case CPUINFO_INT_REGISTER + DSP56K_LC: info->i = LC; break;
case CPUINFO_INT_REGISTER + DSP56K_LA: info->i = LA; break;
case CPUINFO_INT_SP:
case CPUINFO_INT_REGISTER + DSP56K_SP: info->i = SP; break;
case CPUINFO_INT_REGISTER + DSP56K_OMR: info->i = OMR; break;
case CPUINFO_INT_REGISTER + DSP56K_X: info->i = X; break;
case CPUINFO_INT_REGISTER + DSP56K_Y: info->i = Y; break;
case CPUINFO_INT_REGISTER + DSP56K_A: info->i = A; break;
case CPUINFO_INT_REGISTER + DSP56K_B: info->i = B; break;
case CPUINFO_INT_REGISTER + DSP56K_R0: info->i = R0; break;
case CPUINFO_INT_REGISTER + DSP56K_R1: info->i = R1; break;
case CPUINFO_INT_REGISTER + DSP56K_R2: info->i = R2; break;
case CPUINFO_INT_REGISTER + DSP56K_R3: info->i = R3; break;
case CPUINFO_INT_REGISTER + DSP56K_N0: info->i = N0; break;
case CPUINFO_INT_REGISTER + DSP56K_N1: info->i = N1; break;
case CPUINFO_INT_REGISTER + DSP56K_N2: info->i = N2; break;
case CPUINFO_INT_REGISTER + DSP56K_N3: info->i = N3; break;
case CPUINFO_INT_REGISTER + DSP56K_M0: info->i = M0; break;
case CPUINFO_INT_REGISTER + DSP56K_M1: info->i = M1; break;
case CPUINFO_INT_REGISTER + DSP56K_M2: info->i = M2; break;
case CPUINFO_INT_REGISTER + DSP56K_M3: info->i = M3; break;
/* case CPUINFO_INT_REGISTER + DSP56K_TEMP: info->i = TEMP; break; */
/* case CPUINFO_INT_REGISTER + DSP56K_STATUS: info->i = STATUS; break; */
// The CPU stack
case CPUINFO_INT_REGISTER + DSP56K_ST0: info->i = ST0; break;
case CPUINFO_INT_REGISTER + DSP56K_ST1: info->i = ST1; break;
case CPUINFO_INT_REGISTER + DSP56K_ST2: info->i = ST2; break;
case CPUINFO_INT_REGISTER + DSP56K_ST3: info->i = ST3; break;
case CPUINFO_INT_REGISTER + DSP56K_ST4: info->i = ST4; break;
case CPUINFO_INT_REGISTER + DSP56K_ST5: info->i = ST5; break;
case CPUINFO_INT_REGISTER + DSP56K_ST6: info->i = ST6; break;
case CPUINFO_INT_REGISTER + DSP56K_ST7: info->i = ST7; break;
case CPUINFO_INT_REGISTER + DSP56K_ST8: info->i = ST8; break;
case CPUINFO_INT_REGISTER + DSP56K_ST9: info->i = ST9; break;
case CPUINFO_INT_REGISTER + DSP56K_ST10: info->i = ST10; break;
case CPUINFO_INT_REGISTER + DSP56K_ST11: info->i = ST11; break;
case CPUINFO_INT_REGISTER + DSP56K_ST12: info->i = ST12; break;
case CPUINFO_INT_REGISTER + DSP56K_ST13: info->i = ST13; break;
case CPUINFO_INT_REGISTER + DSP56K_ST14: info->i = ST14; break;
case CPUINFO_INT_REGISTER + DSP56K_ST15: info->i = ST15; break;
// --- the following bits of info are returned as pointers to data or functions ---
case CPUINFO_PTR_SET_INFO: info->setinfo = CPU_SET_INFO_NAME(dsp56k); break;
case CPUINFO_PTR_GET_CONTEXT: info->getcontext = CPU_GET_CONTEXT_NAME(dsp56k); break;
case CPUINFO_PTR_SET_CONTEXT: info->setcontext = CPU_SET_CONTEXT_NAME(dsp56k); break;
case CPUINFO_PTR_INIT: info->init = CPU_INIT_NAME(dsp56k); break;
case CPUINFO_PTR_RESET: info->reset = CPU_RESET_NAME(dsp56k); break;
case CPUINFO_PTR_EXIT: info->exit = CPU_EXIT_NAME(dsp56k); break;
case CPUINFO_PTR_EXECUTE: info->execute = CPU_EXECUTE_NAME(dsp56k); break;
case CPUINFO_PTR_BURN: info->burn = NULL; break;
case CPUINFO_PTR_DISASSEMBLE: info->disassemble = CPU_DISASSEMBLE_NAME(dsp56k); break;
case CPUINFO_PTR_DEBUG_INIT: info->debug_init = NULL; break;
case CPUINFO_PTR_TRANSLATE: info->translate = NULL; break;
case CPUINFO_PTR_READ: info->read = NULL; break;
case CPUINFO_PTR_WRITE: info->write = NULL; break;
case CPUINFO_PTR_READOP: info->readop = NULL; break;
case CPUINFO_PTR_INSTRUCTION_COUNTER: info->icount = &dsp56k_icount; break;
case CPUINFO_PTR_INTERNAL_MEMORY_MAP + ADDRESS_SPACE_DATA:
info->internal_map16 = ADDRESS_MAP_NAME(dsp56156_x_data_map); break;
case CPUINFO_PTR_INTERNAL_MEMORY_MAP + ADDRESS_SPACE_PROGRAM:
info->internal_map16 = ADDRESS_MAP_NAME(dsp56156_program_map); break;
// --- the following bits of info are returned as NULL-terminated strings ---
case CPUINFO_STR_NAME: strcpy(info->s, "DSP56156"); break;
case CPUINFO_STR_CORE_FAMILY: strcpy(info->s, "Motorola DSP56156"); break;
case CPUINFO_STR_CORE_VERSION: strcpy(info->s, "0.1"); break;
case CPUINFO_STR_CORE_FILE: strcpy(info->s, __FILE__); break;
case CPUINFO_STR_CORE_CREDITS: strcpy(info->s, "Andrew Gardner"); break;
case CPUINFO_STR_FLAGS:
sprintf(info->s, "%s%s %s%s%s%s%s%s%s%s %s%s",
/* Status Register */
LF_bit() ? "L" : ".",
FV_bit() ? "F" : ".",
S_bit() ? "S" : ".",
L_bit() ? "L" : ".",
E_bit() ? "E" : ".",
U_bit() ? "U" : ".",
N_bit() ? "N" : ".",
Z_bit() ? "Z" : ".",
V_bit() ? "V" : ".",
C_bit() ? "C" : ".",
/* Stack Pointer */
UF_bit() ? "U" : ".",
SE_bit() ? "S" : ".");
break;
case CPUINFO_STR_REGISTER + DSP56K_PC: sprintf(info->s, "PC : %04x", PC); break;
case CPUINFO_STR_REGISTER + DSP56K_SR: sprintf(info->s, "SR : %04x", SR); break;
case CPUINFO_STR_REGISTER + DSP56K_LC: sprintf(info->s, "LC : %04x", LC); break;
case CPUINFO_STR_REGISTER + DSP56K_LA: sprintf(info->s, "LA : %04x", LA); break;
case CPUINFO_STR_REGISTER + DSP56K_SP: sprintf(info->s, "SP : %02x", SP); break;
case CPUINFO_STR_REGISTER + DSP56K_OMR: sprintf(info->s, "OMR: %02x", OMR); break;
case CPUINFO_STR_REGISTER + DSP56K_X: sprintf(info->s, "X : %04x %04x", X1, X0); break;
case CPUINFO_STR_REGISTER + DSP56K_Y: sprintf(info->s, "Y : %04x %04x", Y1, Y0); break;
case CPUINFO_STR_REGISTER + DSP56K_A: sprintf(info->s, "A : %02x %04x %04x", A2,A1,A0); break;
case CPUINFO_STR_REGISTER + DSP56K_B: sprintf(info->s, "B : %02x %04x %04x", B2,B1,B0); break;
case CPUINFO_STR_REGISTER + DSP56K_R0: sprintf(info->s, "R0 : %04x", R0); break;
case CPUINFO_STR_REGISTER + DSP56K_R1: sprintf(info->s, "R1 : %04x", R1); break;
case CPUINFO_STR_REGISTER + DSP56K_R2: sprintf(info->s, "R2 : %04x", R2); break;
case CPUINFO_STR_REGISTER + DSP56K_R3: sprintf(info->s, "R3 : %04x", R3); break;
case CPUINFO_STR_REGISTER + DSP56K_N0: sprintf(info->s, "N0 : %04x", N0); break;
case CPUINFO_STR_REGISTER + DSP56K_N1: sprintf(info->s, "N1 : %04x", N1); break;
case CPUINFO_STR_REGISTER + DSP56K_N2: sprintf(info->s, "N2 : %04x", N2); break;
case CPUINFO_STR_REGISTER + DSP56K_N3: sprintf(info->s, "N3 : %04x", N3); break;
case CPUINFO_STR_REGISTER + DSP56K_M0: sprintf(info->s, "M0 : %04x", M0); break;
case CPUINFO_STR_REGISTER + DSP56K_M1: sprintf(info->s, "M1 : %04x", M1); break;
case CPUINFO_STR_REGISTER + DSP56K_M2: sprintf(info->s, "M2 : %04x", M2); break;
case CPUINFO_STR_REGISTER + DSP56K_M3: sprintf(info->s, "M3 : %04x", M3); break;
/* case CPUINFO_STR_REGISTER + DSP56K_TEMP: sprintf(info->s, "TMP: %04x", TEMP); break; */
/* case CPUINFO_STR_REGISTER + DSP56K_STATUS: sprintf(info->s, "STS: %02x", STATUS); break; */
// The CPU stack
case CPUINFO_STR_REGISTER + DSP56K_ST0: sprintf(info->s, "ST0 : %08x", ST0); break;
case CPUINFO_STR_REGISTER + DSP56K_ST1: sprintf(info->s, "ST1 : %08x", ST1); break;
case CPUINFO_STR_REGISTER + DSP56K_ST2: sprintf(info->s, "ST2 : %08x", ST2); break;
case CPUINFO_STR_REGISTER + DSP56K_ST3: sprintf(info->s, "ST3 : %08x", ST3); break;
case CPUINFO_STR_REGISTER + DSP56K_ST4: sprintf(info->s, "ST4 : %08x", ST4); break;
case CPUINFO_STR_REGISTER + DSP56K_ST5: sprintf(info->s, "ST5 : %08x", ST5); break;
case CPUINFO_STR_REGISTER + DSP56K_ST6: sprintf(info->s, "ST6 : %08x", ST6); break;
case CPUINFO_STR_REGISTER + DSP56K_ST7: sprintf(info->s, "ST7 : %08x", ST7); break;
case CPUINFO_STR_REGISTER + DSP56K_ST8: sprintf(info->s, "ST8 : %08x", ST8); break;
case CPUINFO_STR_REGISTER + DSP56K_ST9: sprintf(info->s, "ST9 : %08x", ST9); break;
case CPUINFO_STR_REGISTER + DSP56K_ST10: sprintf(info->s, "ST10: %08x", ST10); break;
case CPUINFO_STR_REGISTER + DSP56K_ST11: sprintf(info->s, "ST11: %08x", ST11); break;
case CPUINFO_STR_REGISTER + DSP56K_ST12: sprintf(info->s, "ST12: %08x", ST12); break;
case CPUINFO_STR_REGISTER + DSP56K_ST13: sprintf(info->s, "ST13: %08x", ST13); break;
case CPUINFO_STR_REGISTER + DSP56K_ST14: sprintf(info->s, "ST14: %08x", ST14); break;
case CPUINFO_STR_REGISTER + DSP56K_ST15: sprintf(info->s, "ST15: %08x", ST15); break;
}
}