mirror of
https://github.com/holub/mame
synced 2025-05-25 23:35:26 +03:00
2464 lines
93 KiB
C
2464 lines
93 KiB
C
/**************************************************************************\
|
|
* Texas Instruments TMS320x25 DSP Emulator *
|
|
* *
|
|
* Copyright Tony La Porta *
|
|
* Written for the MAME project. *
|
|
* *
|
|
* *
|
|
* Three versions of the chip are available, and they are: *
|
|
* TMS320C25 Internal ROM one time programmed at TI *
|
|
* TMS320E25 Internal ROM programmable as a normal EPROM *
|
|
* TMS320P25 Internal ROM programmable once as a normal EPROM only *
|
|
* These devices can also be used as a MicroController with external ROM *
|
|
* *
|
|
* *
|
|
* Notes : The term 'DMA' within this document, is in reference *
|
|
* to Direct Memory Addressing, and NOT the usual term *
|
|
* of Direct Memory Access. *
|
|
* This is a word based microcontroller, with addressing *
|
|
* architecture based on the Harvard addressing scheme. *
|
|
* *
|
|
* *
|
|
* *
|
|
* **** Change Log **** *
|
|
* *
|
|
* TLP (2x-May-2001) *
|
|
* - Work began on this emulator *
|
|
* TLP (12-Jul-2001) *
|
|
* - First private release *
|
|
* TLP (xx-Dec-2001) Ver 0.11 *
|
|
* - Various undocumented fixes *
|
|
* TLP (13-Jul-2002) Ver 0.12 *
|
|
* - Corrected IRQ2 vector pointer *
|
|
* - Fixed the signedness in many equation based instructions *
|
|
* - Adjusted the level sensing for the Signal inputs *
|
|
* - Added the ability to view the CPU in the debugger when it's halted *
|
|
* TLP (16-Nov-2002) *
|
|
* - First public release after nearly 1.5 years! *
|
|
* - Adjusted more signedness instructions (ADDH, SUBC, SUBH, etc) *
|
|
* TLP (21-Dec-2002) *
|
|
* - Added memory banking for the CNFD, CNFP and CONF instructions *
|
|
* - Corrected IRQ masking checks *
|
|
* TLP (25-Dec-2002) Ver 1.10 *
|
|
* - Added internal timer *
|
|
* *
|
|
\**************************************************************************/
|
|
|
|
/*****************************************************************************
|
|
To fix, or currently lacking from this emulator are:
|
|
|
|
Fix the levels for S_IN and S_OUT - use assert/release line
|
|
|
|
# Support for the built in Timer/Counter Page 91
|
|
When idling, Counter must still be activly counting down. When counter
|
|
reaches 0 it should issue a TINT (if it's not masked), then come out of
|
|
IDLE mode.
|
|
If TINT is masked, the Timer still needs to count down.
|
|
|
|
# Support for the built in Serial Port
|
|
# Support for the Global memory register
|
|
# Support for the switch for RAM block 0 banking between RAM and ROM space
|
|
# Correct the multi-cycle instruction cycle counts
|
|
# Add support to set ROM & RAM as Internal/External in order to correctly
|
|
compute cycle timings
|
|
# Check (read) Hold signal level during execution loop ?
|
|
# Fix bugs
|
|
# Fix more bugs :-)
|
|
# Add/fix other things I forgot
|
|
*****************************************************************************/
|
|
|
|
/*
|
|
TMS32025 CONF Mode Decoding Table
|
|
|=======================================|
|
|
| Status bit | Blocks |
|
|
| CNF | B0 | B1 | B2 |
|
|
|------------+---------+---------+------|
|
|
| 0 0 | data | data | data |
|
|
| 1 1 | program | data | data |
|
|
|=======================================|
|
|
|
|
|
|
TMS32026 CONF Mode Decoding Table
|
|
|==================================================|
|
|
| Status bits | Blocks |
|
|
| CNF1 | CNF0 | B0 | B1 | B2 | B3 |
|
|
|------+------+---------+---------+------+---------|
|
|
| 0 | 0 | data | data | data | data |
|
|
| 0 | 1 | program | data | data | data |
|
|
| 1 | 0 | program | program | data | data |
|
|
| 1 | 1 | program | program | data | program |
|
|
|==================================================|
|
|
|
|
|
|
|
|
Table 3-2. TMS32025/26 Memory Blocks
|
|
|=========================================================|
|
|
| Configured As Data Memory |
|
|
|-------+-------TMS320C25--------+-------TMS320C26--------|
|
|
| | | Hexadecimal | | Hexadecimal |
|
|
| Block | Pages | Address | Pages | Address |
|
|
|-------+---------+--------------+---------+--------------|
|
|
| B2 | 0 | 0060h-007Fh | 0 | 0060h-007Fh |
|
|
| B0 | 4-5 | 0200h-02FFh | 4-7 | 0200h-03FFh |
|
|
| B1 | 6-7 | 0300h-03FFh | 8-11 | 0400h-05FFh |
|
|
| B3 | B3 does not exist | 12-15 | 0600h-07FFh |
|
|
|=========================================================|
|
|
| Configured As Program Memory |
|
|
|-------+-------TMS320C25--------+-------TMS320C26--------|
|
|
| | | Hexadecimal | | Hexadecimal |
|
|
| Block | Pages | Address | Pages | Address |
|
|
|-------+---------+--------------+---------+--------------|
|
|
| B2 | B2 is not configurable | B2 is not configurable |
|
|
| B0 | 510-511 | FF00h-FFFFh | 500-503 | FA00h-FBFFh |
|
|
| B1 | B1 is not configurable | 504-507 | FC00h-FDFFh |
|
|
| B3 | B3 does not exist | 508-511 | FE00h-FFFFh |
|
|
|=========================================================|
|
|
*/
|
|
|
|
|
|
#include "emu.h"
|
|
#include "debugger.h"
|
|
#include "tms32025.h"
|
|
|
|
|
|
#define CLK 4 /* 1 cycle equals 4 clock ticks */ /* PE/DI */
|
|
|
|
|
|
|
|
#define SET_PC(x) do { cpustate->PC = (x); } while (0)
|
|
|
|
#define P_IN(A) (cpustate->io->read_word((A)<<1))
|
|
#define P_OUT(A,V) (cpustate->io->write_word(((A)<<1),(V)))
|
|
#define S_IN(A) (cpustate->io->read_word((A)<<1))
|
|
#define S_OUT(A,V) (cpustate->io->write_word(((A)<<1),(V)))
|
|
|
|
#define M_RDOP(A) ((cpustate->pgmmap[(A) >> 7]) ? (cpustate->pgmmap[(A) >> 7][(A) & 0x7f]) : cpustate->direct->read_decrypted_word((A)<<1))
|
|
#define M_RDOP_ARG(A) ((cpustate->pgmmap[(A) >> 7]) ? (cpustate->pgmmap[(A) >> 7][(A) & 0x7f]) : cpustate->direct->read_decrypted_word((A)<<1))
|
|
|
|
|
|
|
|
typedef struct _tms32025_state tms32025_state; /* Page 3-6 (45) shows all registers */
|
|
struct _tms32025_state
|
|
|
|
{
|
|
/******************** CPU Internal Registers *******************/
|
|
UINT16 PREVPC; /* previous program counter */
|
|
UINT16 PC;
|
|
UINT16 PFC;
|
|
UINT16 STR0, STR1;
|
|
UINT8 IFR;
|
|
UINT8 RPTC;
|
|
PAIR ACC;
|
|
PAIR Preg;
|
|
UINT16 Treg;
|
|
UINT16 AR[8];
|
|
UINT16 STACK[8];
|
|
PAIR ALU;
|
|
UINT16 *intRAM;
|
|
UINT8 timerover;
|
|
|
|
/********************** Status data ****************************/
|
|
PAIR opcode;
|
|
int idle;
|
|
int hold;
|
|
int external_mem_access; /** required for hold mode. Implement it ! */
|
|
int init_load_addr; /* 0=No, 1=Yes, 2=Once for repeat mode */
|
|
int tms32025_irq_cycles;
|
|
int tms32025_dec_cycles;
|
|
device_irq_acknowledge_callback irq_callback;
|
|
|
|
PAIR oldacc;
|
|
UINT32 memaccess;
|
|
int icount;
|
|
int mHackIgnoreARP; /* special handling for lst, lst1 instructions */
|
|
int waiting_for_serial_frame;
|
|
|
|
legacy_cpu_device *device;
|
|
address_space *program;
|
|
direct_read_data *direct;
|
|
address_space *data;
|
|
address_space *io;
|
|
|
|
UINT16 *pgmmap[0x200];
|
|
UINT16 *datamap[0x200];
|
|
};
|
|
|
|
INLINE tms32025_state *get_safe_token(device_t *device)
|
|
{
|
|
assert(device != NULL);
|
|
assert(device->type() == TMS32025 ||
|
|
device->type() == TMS32026);
|
|
return (tms32025_state *)downcast<legacy_cpu_device *>(device)->token();
|
|
}
|
|
|
|
/* opcode table entry */
|
|
typedef struct _tms32025_opcode tms32025_opcode;
|
|
struct _tms32025_opcode
|
|
{
|
|
UINT8 cycles;
|
|
void (*function)(tms32025_state *);
|
|
};
|
|
/* opcode table entry (Opcode CE has sub-opcodes) */
|
|
typedef struct _tms32025_opcode_CE tms32025_opcode_CE;
|
|
struct _tms32025_opcode_CE
|
|
{
|
|
UINT8 cycles;
|
|
void (*function)(tms32025_state *);
|
|
};
|
|
/* opcode table entry (Opcode Dx has sub-opcodes) */
|
|
typedef struct _tms32025_opcode_Dx tms32025_opcode_Dx;
|
|
struct _tms32025_opcode_Dx
|
|
{
|
|
UINT8 cycles;
|
|
void (*function)(tms32025_state *);
|
|
};
|
|
|
|
|
|
|
|
/************************** Memory mapped registers ****************/
|
|
#define DRR cpustate->intRAM[0]
|
|
#define DXR cpustate->intRAM[1]
|
|
#define TIM cpustate->intRAM[2]
|
|
#define PRD cpustate->intRAM[3]
|
|
#define IMR cpustate->intRAM[4]
|
|
#define GREG cpustate->intRAM[5]
|
|
|
|
|
|
|
|
/****************************************************************************
|
|
******* The following is the Status (Flag) register 0 definition. ********
|
|
| 15 | 14 | 13 | 12 | 11 | 10 | 9 | 8 | 7 | 6 | 5 | 4 | 3 | 2 | 1 | 0 |
|
|
| <----ARP---> | OV | OVM | 1 | INTM | <--------------DP---------------> | */
|
|
|
|
#define ARP_REG 0xe000 /* ARP (Auxiliary Register Pointer) */
|
|
#define OV_FLAG 0x1000 /* OV (Overflow flag) 1 indicates an overflow */
|
|
#define OVM_FLAG 0x0800 /* OVM (Overflow Mode bit) 1 forces ACC overflow to greatest positive or negative saturation value */
|
|
#define INTM_FLAG 0x0200 /* INTM (Interrupt Mask flag) 0 enables maskable interrupts */
|
|
#define DP_REG 0x01ff /* DP (Data bank memory Pointer) */
|
|
|
|
|
|
/***********************************************************************************
|
|
*** The following is the Status (Flag) register 1 definition for TMS32025. ********
|
|
| 15 | 14 | 13 | 12 | 11 | 10 | 9 | 8 | 7 | 6 | 5 | 4 | 3 | 2 | 1 | 0 |
|
|
| <----ARB---> | CNF0 | TC | SXM | C | 1 | 1 | HM | FSM | XF | FO | TXM | <-PM-> | */
|
|
|
|
/*** The following is the Status (Flag) register 1 definition for TMS32026. ***********
|
|
| 15 | 14 | 13 | 12 | 11 | 10 | 9 | 8 | 7 | 6 | 5 | 4 | 3 | 2 | 1 | 0 |
|
|
| <----ARB---> | CNF0 | TC | SXM | C | 1 | CNF1 | HM | FSM | XF | FO | TXM | <-PM-> | */
|
|
|
|
#define ARB_REG 0xe000 /* ARB (Auxiliary Register pointer Backup) */
|
|
#define CNF0_REG 0x1000 /* CNF0 (Onchip RAM CoNFiguration) 0 means B0=data memory, 1means B0=program memory */
|
|
#define CNF1_REG 0x0080 /* CNF1 (Onchip RAM CoNFiguration) 0 means B0=data memory, 1means B0=program memory */
|
|
#define TC_FLAG 0x0800 /* TC (Test Control flag) */
|
|
#define SXM_FLAG 0x0400 /* SXM (Sign eXtension Mode) */
|
|
#define C_FLAG 0x0200 /* C (Carry flag) */
|
|
#define HM_FLAG 0x0040 /* HM (Processor Hold Mode) */
|
|
#define FSM_FLAG 0x0020 /* FSM (Frame Synchronization Mode - for serial port) */
|
|
#define XF_FLAG 0x0010 /* XF (XF output pin status) */
|
|
#define FO_FLAG 0x0008 /* FO (Serial port Format In/Out mode) */
|
|
#define TXM_FLAG 0x0004 /* TXM (Transmit Mode - for serial port) */
|
|
#define PM_REG 0x0003 /* PM (Product shift Mode) */
|
|
|
|
|
|
#define OV ( cpustate->STR0 & OV_FLAG) /* OV (Overflow flag) */
|
|
#define OVM ( cpustate->STR0 & OVM_FLAG) /* OVM (Overflow Mode bit) 1 indicates an overflow */
|
|
#define INTM ( cpustate->STR0 & INTM_FLAG) /* INTM (Interrupt enable flag) 0 enables maskable interrupts */
|
|
#define ARP ((cpustate->STR0 & ARP_REG) >> 13) /* ARP (Auxiliary Register Pointer) */
|
|
#define DP ((cpustate->STR0 & DP_REG) << 7) /* DP (Data memory Pointer bit) */
|
|
#define ARB ( cpustate->STR1 & ARB_REG) /* ARB (Backup Auxiliary Register pointer) */
|
|
#define CNF0 ( cpustate->STR1 & CNF0_REG) /* CNF0 (Onchip Ram Config register) */
|
|
#define TC ( cpustate->STR1 & TC_FLAG) /* TC (Test Control Flag) */
|
|
#define SXM ( cpustate->STR1 & SXM_FLAG) /* SXM (Sign Extension Mode) */
|
|
#define CARRY ( cpustate->STR1 & C_FLAG) /* C (Carry Flag for accumulator) */
|
|
#define HM ( cpustate->STR1 & HM_FLAG) /* HM (Processor Hold Mode) */
|
|
#define FSM ( cpustate->STR1 & FSM_FLAG) /* FSM (Frame Synchronization Mode - for serial port) */
|
|
#define XF ( cpustate->STR1 & FSM_FLAG) /* XF (XF output pin status) */
|
|
#define FO ( cpustate->STR1 & FO_FLAG) /* FO (Serial port Format In/Out mode) */
|
|
#define TXM ( cpustate->STR1 & TXM_FLAG) /* TXM (Transmit Mode - for serial port) */
|
|
#define PM ( cpustate->STR1 & PM_REG) /* PM (P register shift Mode. See SHIFT_Preg_TO_ALU below )*/
|
|
|
|
#define DMA (DP | (cpustate->opcode.b.l & 0x7f)) /* address used in direct memory access operations */
|
|
#define DMApg0 (cpustate->opcode.b.l & 0x7f) /* address used in direct memory access operations for sst instruction */
|
|
#define IND cpustate->AR[ARP] /* address used in indirect memory access operations */
|
|
|
|
INLINE void CLR0(tms32025_state *cpustate, UINT16 flag) { cpustate->STR0 &= ~flag; cpustate->STR0 |= 0x0400; }
|
|
INLINE void SET0(tms32025_state *cpustate, UINT16 flag) { cpustate->STR0 |= flag; cpustate->STR0 |= 0x0400; }
|
|
INLINE void CLR1(tms32025_state *cpustate, UINT16 flag) { cpustate->STR1 &= ~flag; cpustate->STR1 |= 0x0180; }
|
|
INLINE void SET1(tms32025_state *cpustate, UINT16 flag) { cpustate->STR1 |= flag; cpustate->STR1 |= 0x0180; }
|
|
|
|
INLINE void MODIFY_DP(tms32025_state *cpustate, int data)
|
|
{
|
|
cpustate->STR0 &= ~DP_REG;
|
|
cpustate->STR0 |= (data & DP_REG);
|
|
cpustate->STR0 |= 0x0400;
|
|
}
|
|
INLINE void MODIFY_PM(tms32025_state *cpustate, int data)
|
|
{
|
|
cpustate->STR1 &= ~PM_REG;
|
|
cpustate->STR1 |= (data & PM_REG);
|
|
cpustate->STR1 |= 0x0180;
|
|
}
|
|
INLINE void MODIFY_ARP(tms32025_state *cpustate, int data)
|
|
{
|
|
cpustate->STR1 &= ~ARB_REG;
|
|
cpustate->STR1 |= (cpustate->STR0 & ARP_REG);
|
|
cpustate->STR1 |= 0x0180;
|
|
cpustate->STR0 &= ~ARP_REG;
|
|
cpustate->STR0 |= ((data << 13) & ARP_REG);
|
|
cpustate->STR0 |= 0x0400;
|
|
}
|
|
|
|
|
|
#ifdef UNUSED_FUNCTION
|
|
INLINE void MODIFY_ARB(tms32025_state *cpustate, int data)
|
|
{
|
|
cpustate->STR1 &= ~ARB_REG;
|
|
cpustate->STR1 |= ((data << 13) & ARB_REG);
|
|
cpustate->STR1 |= 0x0180;
|
|
}
|
|
#endif
|
|
|
|
|
|
INLINE UINT16 M_RDROM(tms32025_state *cpustate, offs_t addr)
|
|
{
|
|
UINT16 *ram;
|
|
addr &= 0xffff;
|
|
ram = cpustate->pgmmap[addr >> 7];
|
|
if (ram) return ram[addr & 0x7f];
|
|
return cpustate->program->read_word(addr << 1);
|
|
}
|
|
|
|
INLINE void M_WRTROM(tms32025_state *cpustate, offs_t addr, UINT16 data)
|
|
{
|
|
UINT16 *ram;
|
|
addr &= 0xffff;
|
|
ram = cpustate->pgmmap[addr >> 7];
|
|
if (ram) { ram[addr & 0x7f] = data; }
|
|
else cpustate->program->write_word(addr << 1, data);
|
|
}
|
|
|
|
INLINE UINT16 M_RDRAM(tms32025_state *cpustate, offs_t addr)
|
|
{
|
|
UINT16 *ram;
|
|
addr &= 0xffff;
|
|
ram = cpustate->datamap[addr >> 7];
|
|
if (ram) return ram[addr & 0x7f];
|
|
return cpustate->data->read_word(addr << 1);
|
|
}
|
|
|
|
INLINE void M_WRTRAM(tms32025_state *cpustate, offs_t addr, UINT16 data)
|
|
{
|
|
UINT16 *ram;
|
|
addr &= 0xffff;
|
|
ram = cpustate->datamap[addr >> 7];
|
|
if (ram) {
|
|
ram[addr & 0x7f] = data;
|
|
if(addr == 1 && ram == cpustate->intRAM && TXM) {
|
|
if(FSM)
|
|
cpustate->waiting_for_serial_frame = 1;
|
|
else
|
|
cpustate->IFR |= 0x20;
|
|
}
|
|
}
|
|
else cpustate->data->write_word(addr << 1, data);
|
|
}
|
|
|
|
|
|
static UINT16 reverse_carry_add(UINT16 arg0, UINT16 arg1 )
|
|
{
|
|
UINT16 result = 0;
|
|
int carry = 0;
|
|
int count;
|
|
for( count=0; count<16; count++ )
|
|
{
|
|
int sum = (arg0>>15)+(arg1>>15)+carry;
|
|
result = (result<<1)|(sum&1);
|
|
carry = sum>>1;
|
|
arg0<<=1;
|
|
arg1<<=1;
|
|
}
|
|
return result;
|
|
}
|
|
|
|
INLINE void MODIFY_AR_ARP(tms32025_state *cpustate)
|
|
{ /* modify address register referenced by ARP */
|
|
switch (cpustate->opcode.b.l & 0x70) /* Cases ordered by predicted useage */
|
|
{
|
|
case 0x00: /* 000 nop */
|
|
break;
|
|
|
|
case 0x10: /* 001 *- */
|
|
cpustate->AR[ARP] -- ;
|
|
break;
|
|
|
|
case 0x20: /* 010 *+ */
|
|
cpustate->AR[ARP] ++ ;
|
|
break;
|
|
|
|
case 0x30: /* 011 reserved */
|
|
break;
|
|
|
|
case 0x40: /* 100 *BR0- */
|
|
cpustate->AR[ARP] = reverse_carry_add(cpustate->AR[ARP],-cpustate->AR[0]);
|
|
break;
|
|
|
|
case 0x50: /* 101 *0- */
|
|
cpustate->AR[ARP] -= cpustate->AR[0];
|
|
break;
|
|
|
|
case 0x60: /* 110 *0+ */
|
|
cpustate->AR[ARP] += cpustate->AR[0];
|
|
break;
|
|
|
|
case 0x70: /* 111 *BR0+ */
|
|
cpustate->AR[ARP] += reverse_carry_add(cpustate->AR[ARP],cpustate->AR[0]);
|
|
break;
|
|
}
|
|
|
|
if( !cpustate->mHackIgnoreARP )
|
|
{
|
|
if (cpustate->opcode.b.l & 8)
|
|
{ /* bit 3 determines if new value is loaded into ARP */
|
|
MODIFY_ARP(cpustate, (cpustate->opcode.b.l & 7) );
|
|
}
|
|
}
|
|
}
|
|
|
|
INLINE void CALCULATE_ADD_CARRY(tms32025_state *cpustate)
|
|
{
|
|
if ( (UINT32)(cpustate->oldacc.d) > (UINT32)(cpustate->ACC.d) ) {
|
|
SET1(cpustate, C_FLAG);
|
|
}
|
|
else {
|
|
CLR1(cpustate, C_FLAG);
|
|
}
|
|
}
|
|
|
|
INLINE void CALCULATE_SUB_CARRY(tms32025_state *cpustate)
|
|
{
|
|
if ( (UINT32)(cpustate->oldacc.d) < (UINT32)(cpustate->ACC.d) ) {
|
|
CLR1(cpustate, C_FLAG);
|
|
}
|
|
else {
|
|
SET1(cpustate, C_FLAG);
|
|
}
|
|
}
|
|
|
|
INLINE void CALCULATE_ADD_OVERFLOW(tms32025_state *cpustate, INT32 addval)
|
|
{
|
|
if ((INT32)((cpustate->ACC.d ^ addval) & (cpustate->oldacc.d ^ cpustate->ACC.d)) < 0)
|
|
{
|
|
SET0(cpustate, OV_FLAG);
|
|
if (OVM)
|
|
{
|
|
cpustate->ACC.d = ((INT32)cpustate->oldacc.d < 0) ? 0x80000000 : 0x7fffffff;
|
|
}
|
|
}
|
|
}
|
|
INLINE void CALCULATE_SUB_OVERFLOW(tms32025_state *cpustate, INT32 subval)
|
|
{
|
|
if ((INT32)((cpustate->oldacc.d ^ subval) & (cpustate->oldacc.d ^ cpustate->ACC.d)) < 0)
|
|
{
|
|
SET0(cpustate, OV_FLAG);
|
|
if (OVM)
|
|
{
|
|
cpustate->ACC.d = ((INT32)cpustate->oldacc.d < 0) ? 0x80000000 : 0x7fffffff;
|
|
}
|
|
}
|
|
}
|
|
|
|
INLINE UINT16 POP_STACK(tms32025_state *cpustate)
|
|
{
|
|
UINT16 data = cpustate->STACK[7];
|
|
cpustate->STACK[7] = cpustate->STACK[6];
|
|
cpustate->STACK[6] = cpustate->STACK[5];
|
|
cpustate->STACK[5] = cpustate->STACK[4];
|
|
cpustate->STACK[4] = cpustate->STACK[3];
|
|
cpustate->STACK[3] = cpustate->STACK[2];
|
|
cpustate->STACK[2] = cpustate->STACK[1];
|
|
cpustate->STACK[1] = cpustate->STACK[0];
|
|
return data;
|
|
}
|
|
INLINE void PUSH_STACK(tms32025_state *cpustate, UINT16 data)
|
|
{
|
|
cpustate->STACK[0] = cpustate->STACK[1];
|
|
cpustate->STACK[1] = cpustate->STACK[2];
|
|
cpustate->STACK[2] = cpustate->STACK[3];
|
|
cpustate->STACK[3] = cpustate->STACK[4];
|
|
cpustate->STACK[4] = cpustate->STACK[5];
|
|
cpustate->STACK[5] = cpustate->STACK[6];
|
|
cpustate->STACK[6] = cpustate->STACK[7];
|
|
cpustate->STACK[7] = data;
|
|
}
|
|
|
|
INLINE void SHIFT_Preg_TO_ALU(tms32025_state *cpustate)
|
|
{
|
|
switch(PM) /* PM (in STR1) is the shift mode for Preg */
|
|
{
|
|
case 0: cpustate->ALU.d = cpustate->Preg.d; break;
|
|
case 1: cpustate->ALU.d = (cpustate->Preg.d << 1); break;
|
|
case 2: cpustate->ALU.d = (cpustate->Preg.d << 4); break;
|
|
case 3: cpustate->ALU.d = (cpustate->Preg.d >> 6); if (cpustate->Preg.d & 0x80000000) cpustate->ALU.d |= 0xfc000000; break;
|
|
default: break;
|
|
}
|
|
}
|
|
|
|
INLINE void GETDATA(tms32025_state *cpustate, int shift,int signext)
|
|
{
|
|
if (cpustate->opcode.b.l & 0x80)
|
|
{ /* indirect memory access */
|
|
cpustate->memaccess = IND;
|
|
}
|
|
else
|
|
{ /* direct memory address */
|
|
cpustate->memaccess = DMA;
|
|
}
|
|
|
|
if (cpustate->memaccess >= 0x800)
|
|
{
|
|
cpustate->external_mem_access = 1; /* Pause if hold pin is active */
|
|
}
|
|
else
|
|
{
|
|
cpustate->external_mem_access = 0;
|
|
}
|
|
|
|
cpustate->ALU.d = (UINT16)M_RDRAM(cpustate, cpustate->memaccess);
|
|
if (signext) cpustate->ALU.d = (INT16)cpustate->ALU.d;
|
|
cpustate->ALU.d <<= shift;
|
|
|
|
/* next ARP */
|
|
if (cpustate->opcode.b.l & 0x80) MODIFY_AR_ARP(cpustate);
|
|
}
|
|
|
|
INLINE void PUTDATA(tms32025_state *cpustate, UINT16 data)
|
|
{
|
|
if (cpustate->opcode.b.l & 0x80) {
|
|
if (cpustate->memaccess >= 0x800) cpustate->external_mem_access = 1; /* Pause if hold pin is active */
|
|
else cpustate->external_mem_access = 0;
|
|
|
|
M_WRTRAM(cpustate, IND, data);
|
|
MODIFY_AR_ARP(cpustate);
|
|
}
|
|
else {
|
|
if (cpustate->memaccess >= 0x800) cpustate->external_mem_access = 1; /* Pause if hold pin is active */
|
|
else cpustate->external_mem_access = 0;
|
|
|
|
M_WRTRAM(cpustate, DMA, data);
|
|
}
|
|
}
|
|
INLINE void PUTDATA_SST(tms32025_state *cpustate, UINT16 data)
|
|
{
|
|
if (cpustate->opcode.b.l & 0x80) cpustate->memaccess = IND;
|
|
else cpustate->memaccess = DMApg0;
|
|
|
|
if (cpustate->memaccess >= 0x800) cpustate->external_mem_access = 1; /* Pause if hold pin is active */
|
|
else cpustate->external_mem_access = 0;
|
|
|
|
if (cpustate->opcode.b.l & 0x80) {
|
|
cpustate->opcode.b.l &= 0xf7; /* Stop ARP changes */
|
|
MODIFY_AR_ARP(cpustate);
|
|
}
|
|
M_WRTRAM(cpustate, cpustate->memaccess, data);
|
|
}
|
|
|
|
|
|
/* The following functions are here to fill the void for the */
|
|
/* opcode call functions. These functions are never actually called. */
|
|
static void opcodes_CE(tms32025_state *cpustate) { }
|
|
static void opcodes_Dx(tms32025_state *cpustate) { }
|
|
|
|
static void illegal(tms32025_state *cpustate)
|
|
{
|
|
logerror("TMS32025: PC = %04x, Illegal opcode = %04x\n", (cpustate->PC-1), cpustate->opcode.w.l);
|
|
}
|
|
|
|
static void abst(tms32025_state *cpustate)
|
|
{
|
|
if ( (INT32)(cpustate->ACC.d) < 0 ) {
|
|
cpustate->ACC.d = -cpustate->ACC.d;
|
|
if (cpustate->ACC.d == 0x80000000) {
|
|
SET0(cpustate, OV_FLAG);
|
|
if (OVM) cpustate->ACC.d-- ;
|
|
}
|
|
}
|
|
CLR1(cpustate, C_FLAG);
|
|
}
|
|
static void add(tms32025_state *cpustate)
|
|
{
|
|
cpustate->oldacc.d = cpustate->ACC.d;
|
|
GETDATA(cpustate, (cpustate->opcode.b.h & 0xf), SXM);
|
|
cpustate->ACC.d += cpustate->ALU.d;
|
|
CALCULATE_ADD_OVERFLOW(cpustate, cpustate->ALU.d);
|
|
CALCULATE_ADD_CARRY(cpustate);
|
|
}
|
|
static void addc(tms32025_state *cpustate)
|
|
{
|
|
cpustate->oldacc.d = cpustate->ACC.d;
|
|
GETDATA(cpustate, 0, 0);
|
|
if (CARRY) cpustate->ACC.d++;
|
|
cpustate->ACC.d += cpustate->ALU.d;
|
|
CALCULATE_ADD_OVERFLOW(cpustate, cpustate->ALU.d);
|
|
if (cpustate->ACC.d == cpustate->oldacc.d) {} /* edge case, carry remains same */
|
|
else CALCULATE_ADD_CARRY(cpustate);
|
|
}
|
|
static void addh(tms32025_state *cpustate)
|
|
{
|
|
cpustate->oldacc.d = cpustate->ACC.d;
|
|
GETDATA(cpustate, 0, 0);
|
|
cpustate->ACC.w.h += cpustate->ALU.w.l;
|
|
if ((INT16)((cpustate->ACC.w.h ^ cpustate->ALU.w.l) & (cpustate->oldacc.w.h ^ cpustate->ACC.w.h)) < 0) {
|
|
SET0(cpustate, OV_FLAG);
|
|
if (OVM)
|
|
cpustate->ACC.w.h = ((INT16)cpustate->oldacc.w.h < 0) ? 0x8000 : 0x7fff;
|
|
}
|
|
if ( ((INT16)(cpustate->oldacc.w.h) < 0) && ((INT16)(cpustate->ACC.w.h) >= 0) ) {
|
|
SET1(cpustate, C_FLAG);
|
|
}
|
|
/* Carry flag is not cleared, if no carry occurred */
|
|
}
|
|
static void addk(tms32025_state *cpustate)
|
|
{
|
|
cpustate->oldacc.d = cpustate->ACC.d;
|
|
cpustate->ALU.d = (UINT8)cpustate->opcode.b.l;
|
|
cpustate->ACC.d += cpustate->ALU.d;
|
|
CALCULATE_ADD_OVERFLOW(cpustate, cpustate->ALU.d);
|
|
CALCULATE_ADD_CARRY(cpustate);
|
|
}
|
|
static void adds(tms32025_state *cpustate)
|
|
{
|
|
cpustate->oldacc.d = cpustate->ACC.d;
|
|
GETDATA(cpustate, 0, 0);
|
|
cpustate->ACC.d += cpustate->ALU.d;
|
|
CALCULATE_ADD_OVERFLOW(cpustate, cpustate->ALU.d);
|
|
CALCULATE_ADD_CARRY(cpustate);
|
|
}
|
|
static void addt(tms32025_state *cpustate)
|
|
{
|
|
cpustate->oldacc.d = cpustate->ACC.d;
|
|
GETDATA(cpustate, (cpustate->Treg & 0xf), SXM);
|
|
cpustate->ACC.d += cpustate->ALU.d;
|
|
CALCULATE_ADD_OVERFLOW(cpustate, cpustate->ALU.d);
|
|
CALCULATE_ADD_CARRY(cpustate);
|
|
}
|
|
static void adlk(tms32025_state *cpustate)
|
|
{
|
|
cpustate->oldacc.d = cpustate->ACC.d;
|
|
if (SXM) cpustate->ALU.d = (INT16)M_RDOP_ARG(cpustate->PC);
|
|
else cpustate->ALU.d = (UINT16)M_RDOP_ARG(cpustate->PC);
|
|
cpustate->PC++;
|
|
cpustate->ALU.d <<= (cpustate->opcode.b.h & 0xf);
|
|
cpustate->ACC.d += cpustate->ALU.d;
|
|
CALCULATE_ADD_OVERFLOW(cpustate, cpustate->ALU.d);
|
|
CALCULATE_ADD_CARRY(cpustate);
|
|
}
|
|
static void adrk(tms32025_state *cpustate)
|
|
{
|
|
cpustate->AR[ARP] += cpustate->opcode.b.l;
|
|
}
|
|
static void and_(tms32025_state *cpustate)
|
|
{
|
|
GETDATA(cpustate, 0, 0);
|
|
cpustate->ACC.d &= cpustate->ALU.d;
|
|
}
|
|
static void andk(tms32025_state *cpustate)
|
|
{
|
|
cpustate->oldacc.d = cpustate->ACC.d;
|
|
cpustate->ALU.d = (UINT16)M_RDOP_ARG(cpustate->PC);
|
|
cpustate->PC++;
|
|
cpustate->ALU.d <<= (cpustate->opcode.b.h & 0xf);
|
|
cpustate->ACC.d &= cpustate->ALU.d;
|
|
cpustate->ACC.d &= 0x7fffffff;
|
|
}
|
|
static void apac(tms32025_state *cpustate)
|
|
{
|
|
cpustate->oldacc.d = cpustate->ACC.d;
|
|
SHIFT_Preg_TO_ALU(cpustate);
|
|
cpustate->ACC.d += cpustate->ALU.d;
|
|
CALCULATE_ADD_OVERFLOW(cpustate, cpustate->ALU.d);
|
|
CALCULATE_ADD_CARRY(cpustate);
|
|
}
|
|
static void br(tms32025_state *cpustate)
|
|
{
|
|
SET_PC(M_RDOP_ARG(cpustate->PC));
|
|
MODIFY_AR_ARP(cpustate);
|
|
}
|
|
static void bacc(tms32025_state *cpustate)
|
|
{
|
|
SET_PC(cpustate->ACC.w.l);
|
|
}
|
|
static void banz(tms32025_state *cpustate)
|
|
{
|
|
if (cpustate->AR[ARP]) SET_PC(M_RDOP_ARG(cpustate->PC));
|
|
else cpustate->PC++ ;
|
|
MODIFY_AR_ARP(cpustate);
|
|
}
|
|
static void bbnz(tms32025_state *cpustate)
|
|
{
|
|
if (TC) SET_PC(M_RDOP_ARG(cpustate->PC));
|
|
else cpustate->PC++ ;
|
|
MODIFY_AR_ARP(cpustate);
|
|
}
|
|
static void bbz(tms32025_state *cpustate)
|
|
{
|
|
if (TC == 0) SET_PC(M_RDOP_ARG(cpustate->PC));
|
|
else cpustate->PC++ ;
|
|
MODIFY_AR_ARP(cpustate);
|
|
}
|
|
static void bc(tms32025_state *cpustate)
|
|
{
|
|
if (CARRY) SET_PC(M_RDOP_ARG(cpustate->PC));
|
|
else cpustate->PC++ ;
|
|
MODIFY_AR_ARP(cpustate);
|
|
}
|
|
static void bgez(tms32025_state *cpustate)
|
|
{
|
|
if ( (INT32)(cpustate->ACC.d) >= 0 ) SET_PC(M_RDOP_ARG(cpustate->PC));
|
|
else cpustate->PC++ ;
|
|
MODIFY_AR_ARP(cpustate);
|
|
}
|
|
static void bgz(tms32025_state *cpustate)
|
|
{
|
|
if ( (INT32)(cpustate->ACC.d) > 0 ) SET_PC(M_RDOP_ARG(cpustate->PC));
|
|
else cpustate->PC++ ;
|
|
MODIFY_AR_ARP(cpustate);
|
|
}
|
|
static void bioz(tms32025_state *cpustate)
|
|
{
|
|
if (S_IN(TMS32025_BIO) != CLEAR_LINE) SET_PC(M_RDOP_ARG(cpustate->PC));
|
|
else cpustate->PC++ ;
|
|
MODIFY_AR_ARP(cpustate);
|
|
}
|
|
static void bit(tms32025_state *cpustate)
|
|
{
|
|
GETDATA(cpustate, 0, 0);
|
|
if (cpustate->ALU.d & (0x8000 >> (cpustate->opcode.b.h & 0xf))) SET1(cpustate, TC_FLAG);
|
|
else CLR1(cpustate, TC_FLAG);
|
|
}
|
|
static void bitt(tms32025_state *cpustate)
|
|
{
|
|
GETDATA(cpustate, 0, 0);
|
|
if (cpustate->ALU.d & (0x8000 >> (cpustate->Treg & 0xf))) SET1(cpustate, TC_FLAG);
|
|
else CLR1(cpustate, TC_FLAG);
|
|
}
|
|
static void blez(tms32025_state *cpustate)
|
|
{
|
|
if ( (INT32)(cpustate->ACC.d) <= 0 ) SET_PC(M_RDOP_ARG(cpustate->PC));
|
|
else cpustate->PC++ ;
|
|
MODIFY_AR_ARP(cpustate);
|
|
}
|
|
static void blkd(tms32025_state *cpustate)
|
|
{ /** Fix cycle timing **/
|
|
if (cpustate->init_load_addr) {
|
|
cpustate->PFC = M_RDOP_ARG(cpustate->PC);
|
|
cpustate->PC++;
|
|
}
|
|
cpustate->ALU.d = M_RDRAM(cpustate, cpustate->PFC);
|
|
PUTDATA(cpustate, cpustate->ALU.d);
|
|
cpustate->PFC++;
|
|
cpustate->tms32025_dec_cycles += (1*CLK);
|
|
}
|
|
static void blkp(tms32025_state *cpustate)
|
|
{ /** Fix cycle timing **/
|
|
if (cpustate->init_load_addr) {
|
|
cpustate->PFC = M_RDOP_ARG(cpustate->PC);
|
|
cpustate->PC++;
|
|
}
|
|
cpustate->ALU.d = M_RDROM(cpustate, cpustate->PFC);
|
|
PUTDATA(cpustate, cpustate->ALU.d);
|
|
cpustate->PFC++;
|
|
cpustate->tms32025_dec_cycles += (2*CLK);
|
|
}
|
|
static void blz(tms32025_state *cpustate)
|
|
{
|
|
if ( (INT32)(cpustate->ACC.d) < 0 ) SET_PC(M_RDOP_ARG(cpustate->PC));
|
|
else cpustate->PC++ ;
|
|
MODIFY_AR_ARP(cpustate);
|
|
}
|
|
static void bnc(tms32025_state *cpustate)
|
|
{
|
|
if (CARRY == 0) SET_PC(M_RDOP_ARG(cpustate->PC));
|
|
else cpustate->PC++ ;
|
|
MODIFY_AR_ARP(cpustate);
|
|
}
|
|
static void bnv(tms32025_state *cpustate)
|
|
{
|
|
if (OV == 0) SET_PC(M_RDOP_ARG(cpustate->PC));
|
|
else {
|
|
cpustate->PC++ ;
|
|
CLR0(cpustate, OV_FLAG);
|
|
}
|
|
MODIFY_AR_ARP(cpustate);
|
|
}
|
|
static void bnz(tms32025_state *cpustate)
|
|
{
|
|
if (cpustate->ACC.d != 0) SET_PC(M_RDOP_ARG(cpustate->PC));
|
|
else cpustate->PC++ ;
|
|
MODIFY_AR_ARP(cpustate);
|
|
}
|
|
static void bv(tms32025_state *cpustate)
|
|
{
|
|
if (OV) {
|
|
SET_PC(M_RDOP_ARG(cpustate->PC));
|
|
CLR0(cpustate, OV_FLAG);
|
|
}
|
|
else cpustate->PC++ ;
|
|
MODIFY_AR_ARP(cpustate);
|
|
}
|
|
static void bz(tms32025_state *cpustate)
|
|
{
|
|
if (cpustate->ACC.d == 0) SET_PC(M_RDOP_ARG(cpustate->PC));
|
|
else cpustate->PC++ ;
|
|
MODIFY_AR_ARP(cpustate);
|
|
}
|
|
static void cala(tms32025_state *cpustate)
|
|
{
|
|
PUSH_STACK(cpustate, cpustate->PC);
|
|
SET_PC(cpustate->ACC.w.l);
|
|
}
|
|
static void call(tms32025_state *cpustate)
|
|
{
|
|
cpustate->PC++ ;
|
|
PUSH_STACK(cpustate, cpustate->PC);
|
|
SET_PC(M_RDOP_ARG((cpustate->PC - 1)));
|
|
MODIFY_AR_ARP(cpustate);
|
|
}
|
|
static void cmpl(tms32025_state *cpustate)
|
|
{
|
|
cpustate->ACC.d = (~cpustate->ACC.d);
|
|
}
|
|
static void cmpr(tms32025_state *cpustate)
|
|
{
|
|
switch (cpustate->opcode.b.l & 3)
|
|
{
|
|
case 00: if ( (UINT16)(cpustate->AR[ARP]) == (UINT16)(cpustate->AR[0]) ) SET1(cpustate, TC_FLAG);
|
|
else CLR1(cpustate, TC_FLAG);
|
|
break;
|
|
case 01: if ( (UINT16)(cpustate->AR[ARP]) < (UINT16)(cpustate->AR[0]) ) SET1(cpustate, TC_FLAG);
|
|
else CLR1(cpustate, TC_FLAG);
|
|
break;
|
|
case 02: if ( (UINT16)(cpustate->AR[ARP]) > (UINT16)(cpustate->AR[0]) ) SET1(cpustate, TC_FLAG);
|
|
else CLR1(cpustate, TC_FLAG);
|
|
break;
|
|
case 03: if ( (UINT16)(cpustate->AR[ARP]) != (UINT16)(cpustate->AR[0]) ) SET1(cpustate, TC_FLAG);
|
|
else CLR1(cpustate, TC_FLAG);
|
|
break;
|
|
default: break;
|
|
}
|
|
}
|
|
static void cnfd(tms32025_state *cpustate) /** next two fetches need to use previous CNF value ! **/
|
|
{
|
|
CLR1(cpustate, CNF0_REG);
|
|
cpustate->datamap[4] = &cpustate->intRAM[0x200]; /* B0 */
|
|
cpustate->datamap[5] = &cpustate->intRAM[0x280]; /* B0 */
|
|
cpustate->pgmmap[510] = NULL;
|
|
cpustate->pgmmap[511] = NULL;
|
|
}
|
|
static void cnfp(tms32025_state *cpustate) /** next two fetches need to use previous CNF value ! **/
|
|
{
|
|
SET1(cpustate, CNF0_REG);
|
|
cpustate->datamap[4] = NULL; /* B0 */
|
|
cpustate->datamap[5] = NULL; /* B0 */
|
|
cpustate->pgmmap[510] = &cpustate->intRAM[0x200];
|
|
cpustate->pgmmap[511] = &cpustate->intRAM[0x280];
|
|
}
|
|
static void conf(tms32025_state *cpustate) /** Need to reconfigure the memory blocks */
|
|
{
|
|
switch (cpustate->opcode.b.l & 3)
|
|
{
|
|
case 00: CLR1(cpustate, CNF1_REG); CLR1(cpustate, CNF0_REG);
|
|
cpustate->datamap[4] = &cpustate->intRAM[0x200]; /* B0 */
|
|
cpustate->datamap[5] = &cpustate->intRAM[0x280]; /* B0 */
|
|
cpustate->datamap[6] = &cpustate->intRAM[0x300]; /* B0 */
|
|
cpustate->datamap[7] = &cpustate->intRAM[0x380]; /* B0 */
|
|
cpustate->datamap[8] = &cpustate->intRAM[0x400]; /* B1 */
|
|
cpustate->datamap[9] = &cpustate->intRAM[0x480]; /* B1 */
|
|
cpustate->datamap[10] = &cpustate->intRAM[0x500]; /* B1 */
|
|
cpustate->datamap[11] = &cpustate->intRAM[0x580]; /* B1 */
|
|
cpustate->datamap[12] = &cpustate->intRAM[0x600]; /* B3 */
|
|
cpustate->datamap[13] = &cpustate->intRAM[0x680]; /* B3 */
|
|
cpustate->datamap[14] = &cpustate->intRAM[0x700]; /* B3 */
|
|
cpustate->datamap[15] = &cpustate->intRAM[0x780]; /* B3 */
|
|
cpustate->pgmmap[500] = NULL;
|
|
cpustate->pgmmap[501] = NULL;
|
|
cpustate->pgmmap[502] = NULL;
|
|
cpustate->pgmmap[503] = NULL;
|
|
cpustate->pgmmap[504] = NULL;
|
|
cpustate->pgmmap[505] = NULL;
|
|
cpustate->pgmmap[506] = NULL;
|
|
cpustate->pgmmap[507] = NULL;
|
|
cpustate->pgmmap[508] = NULL;
|
|
cpustate->pgmmap[509] = NULL;
|
|
cpustate->pgmmap[510] = NULL;
|
|
cpustate->pgmmap[511] = NULL;
|
|
break;
|
|
|
|
case 01: CLR1(cpustate, CNF1_REG); SET1(cpustate, CNF0_REG);
|
|
cpustate->datamap[4] = NULL;
|
|
cpustate->datamap[5] = NULL;
|
|
cpustate->datamap[6] = NULL;
|
|
cpustate->datamap[7] = NULL;
|
|
cpustate->datamap[8] = &cpustate->intRAM[0x400]; /* B1 */
|
|
cpustate->datamap[9] = &cpustate->intRAM[0x480]; /* B1 */
|
|
cpustate->datamap[10] = &cpustate->intRAM[0x500]; /* B1 */
|
|
cpustate->datamap[11] = &cpustate->intRAM[0x580]; /* B1 */
|
|
cpustate->datamap[12] = &cpustate->intRAM[0x600]; /* B3 */
|
|
cpustate->datamap[13] = &cpustate->intRAM[0x680]; /* B3 */
|
|
cpustate->datamap[14] = &cpustate->intRAM[0x700]; /* B3 */
|
|
cpustate->datamap[15] = &cpustate->intRAM[0x780]; /* B3 */
|
|
cpustate->pgmmap[500] = &cpustate->intRAM[0x200]; /* B0 */
|
|
cpustate->pgmmap[501] = &cpustate->intRAM[0x280]; /* B0 */
|
|
cpustate->pgmmap[502] = &cpustate->intRAM[0x300]; /* B0 */
|
|
cpustate->pgmmap[503] = &cpustate->intRAM[0x380]; /* B0 */
|
|
cpustate->pgmmap[504] = NULL;
|
|
cpustate->pgmmap[505] = NULL;
|
|
cpustate->pgmmap[506] = NULL;
|
|
cpustate->pgmmap[507] = NULL;
|
|
cpustate->pgmmap[508] = NULL;
|
|
cpustate->pgmmap[509] = NULL;
|
|
cpustate->pgmmap[510] = NULL;
|
|
cpustate->pgmmap[511] = NULL;
|
|
break;
|
|
|
|
case 02: SET1(cpustate, CNF1_REG); CLR1(cpustate, CNF0_REG);
|
|
cpustate->datamap[4] = NULL;
|
|
cpustate->datamap[5] = NULL;
|
|
cpustate->datamap[6] = NULL;
|
|
cpustate->datamap[7] = NULL;
|
|
cpustate->datamap[8] = NULL;
|
|
cpustate->datamap[9] = NULL;
|
|
cpustate->datamap[10] = NULL;
|
|
cpustate->datamap[11] = NULL;
|
|
cpustate->datamap[12] = &cpustate->intRAM[0x600]; /* B3 */
|
|
cpustate->datamap[13] = &cpustate->intRAM[0x680]; /* B3 */
|
|
cpustate->datamap[14] = &cpustate->intRAM[0x700]; /* B3 */
|
|
cpustate->datamap[15] = &cpustate->intRAM[0x780]; /* B3 */
|
|
cpustate->pgmmap[500] = &cpustate->intRAM[0x200]; /* B0 */
|
|
cpustate->pgmmap[501] = &cpustate->intRAM[0x280]; /* B0 */
|
|
cpustate->pgmmap[502] = &cpustate->intRAM[0x300]; /* B0 */
|
|
cpustate->pgmmap[503] = &cpustate->intRAM[0x380]; /* B0 */
|
|
cpustate->pgmmap[504] = &cpustate->intRAM[0x400]; /* B1 */
|
|
cpustate->pgmmap[505] = &cpustate->intRAM[0x480]; /* B1 */
|
|
cpustate->pgmmap[506] = &cpustate->intRAM[0x500]; /* B1 */
|
|
cpustate->pgmmap[507] = &cpustate->intRAM[0x580]; /* B1 */
|
|
cpustate->pgmmap[508] = NULL;
|
|
cpustate->pgmmap[509] = NULL;
|
|
cpustate->pgmmap[510] = NULL;
|
|
cpustate->pgmmap[511] = NULL;
|
|
break;
|
|
|
|
case 03: SET1(cpustate, CNF1_REG); SET1(cpustate, CNF0_REG);
|
|
cpustate->datamap[4] = NULL;
|
|
cpustate->datamap[5] = NULL;
|
|
cpustate->datamap[6] = NULL;
|
|
cpustate->datamap[7] = NULL;
|
|
cpustate->datamap[8] = NULL;
|
|
cpustate->datamap[9] = NULL;
|
|
cpustate->datamap[10] = NULL;
|
|
cpustate->datamap[11] = NULL;
|
|
cpustate->datamap[12] = NULL;
|
|
cpustate->datamap[13] = NULL;
|
|
cpustate->datamap[14] = NULL;
|
|
cpustate->datamap[15] = NULL;
|
|
cpustate->pgmmap[500] = &cpustate->intRAM[0x200]; /* B0 */
|
|
cpustate->pgmmap[501] = &cpustate->intRAM[0x280]; /* B0 */
|
|
cpustate->pgmmap[502] = &cpustate->intRAM[0x300]; /* B0 */
|
|
cpustate->pgmmap[503] = &cpustate->intRAM[0x380]; /* B0 */
|
|
cpustate->pgmmap[504] = &cpustate->intRAM[0x400]; /* B1 */
|
|
cpustate->pgmmap[505] = &cpustate->intRAM[0x480]; /* B1 */
|
|
cpustate->pgmmap[506] = &cpustate->intRAM[0x500]; /* B1 */
|
|
cpustate->pgmmap[507] = &cpustate->intRAM[0x580]; /* B1 */
|
|
cpustate->pgmmap[508] = &cpustate->intRAM[0x600]; /* B3 */
|
|
cpustate->pgmmap[509] = &cpustate->intRAM[0x680]; /* B3 */
|
|
cpustate->pgmmap[510] = &cpustate->intRAM[0x700]; /* B3 */
|
|
cpustate->pgmmap[511] = &cpustate->intRAM[0x780]; /* B3 */
|
|
break;
|
|
|
|
default: break;
|
|
}
|
|
}
|
|
static void dint(tms32025_state *cpustate)
|
|
{
|
|
SET0(cpustate, INTM_FLAG);
|
|
}
|
|
static void dmov(tms32025_state *cpustate) /** Careful with how memory is configured !! */
|
|
{
|
|
GETDATA(cpustate, 0, 0);
|
|
M_WRTRAM(cpustate, (cpustate->memaccess + 1), cpustate->ALU.w.l);
|
|
}
|
|
static void eint(tms32025_state *cpustate)
|
|
{
|
|
CLR0(cpustate, INTM_FLAG);
|
|
}
|
|
static void fort(tms32025_state *cpustate)
|
|
{
|
|
if (cpustate->opcode.b.l & 1) SET1(cpustate, FO_FLAG);
|
|
else CLR1(cpustate, FO_FLAG);
|
|
}
|
|
static void idle(tms32025_state *cpustate)
|
|
{
|
|
CLR0(cpustate, INTM_FLAG);
|
|
cpustate->idle = 1;
|
|
}
|
|
static void in(tms32025_state *cpustate)
|
|
{
|
|
cpustate->ALU.w.l = P_IN( (cpustate->opcode.b.h & 0xf) );
|
|
PUTDATA(cpustate, cpustate->ALU.w.l);
|
|
}
|
|
static void lac(tms32025_state *cpustate)
|
|
{
|
|
GETDATA(cpustate, (cpustate->opcode.b.h & 0xf), SXM);
|
|
cpustate->ACC.d = cpustate->ALU.d;
|
|
}
|
|
static void lack(tms32025_state *cpustate) /* ZAC is a subset of this instruction */
|
|
{
|
|
cpustate->ACC.d = (UINT8)cpustate->opcode.b.l;
|
|
}
|
|
static void lact(tms32025_state *cpustate)
|
|
{
|
|
GETDATA(cpustate, (cpustate->Treg & 0xf), SXM);
|
|
cpustate->ACC.d = cpustate->ALU.d;
|
|
}
|
|
static void lalk(tms32025_state *cpustate)
|
|
{
|
|
if (SXM) {
|
|
cpustate->ALU.d = (INT16)M_RDOP_ARG(cpustate->PC);
|
|
cpustate->ACC.d = cpustate->ALU.d << (cpustate->opcode.b.h & 0xf);
|
|
}
|
|
else {
|
|
cpustate->ALU.d = (UINT16)M_RDOP_ARG(cpustate->PC);
|
|
cpustate->ACC.d = cpustate->ALU.d << (cpustate->opcode.b.h & 0xf);
|
|
cpustate->ACC.d &= 0x7fffffff;
|
|
}
|
|
cpustate->PC++;
|
|
}
|
|
static void lar_ar0(tms32025_state *cpustate) { GETDATA(cpustate, 0, 0); cpustate->AR[0] = cpustate->ALU.w.l; }
|
|
static void lar_ar1(tms32025_state *cpustate) { GETDATA(cpustate, 0, 0); cpustate->AR[1] = cpustate->ALU.w.l; }
|
|
static void lar_ar2(tms32025_state *cpustate) { GETDATA(cpustate, 0, 0); cpustate->AR[2] = cpustate->ALU.w.l; }
|
|
static void lar_ar3(tms32025_state *cpustate) { GETDATA(cpustate, 0, 0); cpustate->AR[3] = cpustate->ALU.w.l; }
|
|
static void lar_ar4(tms32025_state *cpustate) { GETDATA(cpustate, 0, 0); cpustate->AR[4] = cpustate->ALU.w.l; }
|
|
static void lar_ar5(tms32025_state *cpustate) { GETDATA(cpustate, 0, 0); cpustate->AR[5] = cpustate->ALU.w.l; }
|
|
static void lar_ar6(tms32025_state *cpustate) { GETDATA(cpustate, 0, 0); cpustate->AR[6] = cpustate->ALU.w.l; }
|
|
static void lar_ar7(tms32025_state *cpustate) { GETDATA(cpustate, 0, 0); cpustate->AR[7] = cpustate->ALU.w.l; }
|
|
static void lark_ar0(tms32025_state *cpustate) { cpustate->AR[0] = cpustate->opcode.b.l; }
|
|
static void lark_ar1(tms32025_state *cpustate) { cpustate->AR[1] = cpustate->opcode.b.l; }
|
|
static void lark_ar2(tms32025_state *cpustate) { cpustate->AR[2] = cpustate->opcode.b.l; }
|
|
static void lark_ar3(tms32025_state *cpustate) { cpustate->AR[3] = cpustate->opcode.b.l; }
|
|
static void lark_ar4(tms32025_state *cpustate) { cpustate->AR[4] = cpustate->opcode.b.l; }
|
|
static void lark_ar5(tms32025_state *cpustate) { cpustate->AR[5] = cpustate->opcode.b.l; }
|
|
static void lark_ar6(tms32025_state *cpustate) { cpustate->AR[6] = cpustate->opcode.b.l; }
|
|
static void lark_ar7(tms32025_state *cpustate) { cpustate->AR[7] = cpustate->opcode.b.l; }
|
|
static void ldp(tms32025_state *cpustate)
|
|
{
|
|
GETDATA(cpustate, 0, 0);
|
|
MODIFY_DP(cpustate, cpustate->ALU.d & 0x1ff);
|
|
}
|
|
static void ldpk(tms32025_state *cpustate)
|
|
{
|
|
MODIFY_DP(cpustate, cpustate->opcode.w.l & 0x1ff);
|
|
}
|
|
static void lph(tms32025_state *cpustate)
|
|
{
|
|
GETDATA(cpustate, 0, 0);
|
|
cpustate->Preg.w.h = cpustate->ALU.w.l;
|
|
}
|
|
static void lrlk(tms32025_state *cpustate)
|
|
{
|
|
cpustate->ALU.d = (UINT16)M_RDOP_ARG(cpustate->PC);
|
|
cpustate->PC++;
|
|
cpustate->AR[cpustate->opcode.b.h & 7] = cpustate->ALU.w.l;
|
|
}
|
|
static void lst(tms32025_state *cpustate)
|
|
{
|
|
cpustate->mHackIgnoreARP = 1;
|
|
GETDATA(cpustate, 0, 0);
|
|
cpustate->mHackIgnoreARP = 0;
|
|
|
|
cpustate->ALU.w.l &= (~INTM_FLAG);
|
|
cpustate->STR0 &= INTM_FLAG;
|
|
cpustate->STR0 |= cpustate->ALU.w.l; /* Must not affect INTM */
|
|
cpustate->STR0 |= 0x0400;
|
|
}
|
|
static void lst1(tms32025_state *cpustate)
|
|
{
|
|
cpustate->mHackIgnoreARP = 1;
|
|
GETDATA(cpustate, 0, 0);
|
|
cpustate->mHackIgnoreARP = 0;
|
|
|
|
cpustate->STR1 = cpustate->ALU.w.l;
|
|
cpustate->STR1 |= 0x0180;
|
|
cpustate->STR0 &= (~ARP_REG); /* ARB also gets copied to ARP */
|
|
cpustate->STR0 |= (cpustate->STR1 & ARB_REG);
|
|
}
|
|
static void lt(tms32025_state *cpustate)
|
|
{
|
|
GETDATA(cpustate, 0, 0);
|
|
cpustate->Treg = cpustate->ALU.w.l;
|
|
}
|
|
static void lta(tms32025_state *cpustate)
|
|
{
|
|
cpustate->oldacc.d = cpustate->ACC.d;
|
|
GETDATA(cpustate, 0, 0);
|
|
cpustate->Treg = cpustate->ALU.w.l;
|
|
SHIFT_Preg_TO_ALU(cpustate);
|
|
cpustate->ACC.d += cpustate->ALU.d;
|
|
CALCULATE_ADD_OVERFLOW(cpustate, cpustate->ALU.d);
|
|
CALCULATE_ADD_CARRY(cpustate);
|
|
}
|
|
static void ltd(tms32025_state *cpustate) /** Careful with how memory is configured !! */
|
|
{
|
|
cpustate->oldacc.d = cpustate->ACC.d;
|
|
GETDATA(cpustate, 0, 0);
|
|
cpustate->Treg = cpustate->ALU.w.l;
|
|
M_WRTRAM(cpustate, (cpustate->memaccess+1), cpustate->ALU.w.l);
|
|
SHIFT_Preg_TO_ALU(cpustate);
|
|
cpustate->ACC.d += cpustate->ALU.d;
|
|
CALCULATE_ADD_OVERFLOW(cpustate, cpustate->ALU.d);
|
|
CALCULATE_ADD_CARRY(cpustate);
|
|
}
|
|
static void ltp(tms32025_state *cpustate)
|
|
{
|
|
cpustate->oldacc.d = cpustate->ACC.d;
|
|
GETDATA(cpustate, 0, 0);
|
|
cpustate->Treg = cpustate->ALU.w.l;
|
|
SHIFT_Preg_TO_ALU(cpustate);
|
|
cpustate->ACC.d = cpustate->ALU.d;
|
|
}
|
|
static void lts(tms32025_state *cpustate)
|
|
{
|
|
cpustate->oldacc.d = cpustate->ACC.d;
|
|
GETDATA(cpustate, 0, 0);
|
|
cpustate->Treg = cpustate->ALU.w.l;
|
|
SHIFT_Preg_TO_ALU(cpustate);
|
|
cpustate->ACC.d -= cpustate->ALU.d;
|
|
CALCULATE_SUB_OVERFLOW(cpustate, cpustate->ALU.d);
|
|
CALCULATE_SUB_CARRY(cpustate);
|
|
}
|
|
static void mac(tms32025_state *cpustate) /** RAM blocks B0,B1,B2 may be important ! */
|
|
{ /** Fix cycle timing **/
|
|
cpustate->oldacc.d = cpustate->ACC.d;
|
|
if (cpustate->init_load_addr) {
|
|
cpustate->PFC = M_RDOP_ARG(cpustate->PC);
|
|
cpustate->PC++;
|
|
}
|
|
SHIFT_Preg_TO_ALU(cpustate);
|
|
cpustate->ACC.d += cpustate->ALU.d;
|
|
CALCULATE_ADD_OVERFLOW(cpustate, cpustate->ALU.d);
|
|
CALCULATE_ADD_CARRY(cpustate);
|
|
GETDATA(cpustate, 0, 0);
|
|
cpustate->Treg = cpustate->ALU.w.l;
|
|
cpustate->Preg.d = ( (INT16)cpustate->ALU.w.l * (INT16)M_RDROM(cpustate, cpustate->PFC) );
|
|
cpustate->PFC++;
|
|
cpustate->tms32025_dec_cycles += (2*CLK);
|
|
}
|
|
static void macd(tms32025_state *cpustate) /** RAM blocks B0,B1,B2 may be important ! */
|
|
{ /** Fix cycle timing **/
|
|
cpustate->oldacc.d = cpustate->ACC.d;
|
|
if (cpustate->init_load_addr) {
|
|
cpustate->PFC = M_RDOP_ARG(cpustate->PC);
|
|
cpustate->PC++;
|
|
}
|
|
SHIFT_Preg_TO_ALU(cpustate);
|
|
cpustate->ACC.d += cpustate->ALU.d;
|
|
CALCULATE_ADD_OVERFLOW(cpustate, cpustate->ALU.d);
|
|
CALCULATE_ADD_CARRY(cpustate);
|
|
GETDATA(cpustate, 0, 0);
|
|
if ( (cpustate->opcode.b.l & 0x80) || cpustate->init_load_addr ) { /* No writing during repitition, or DMA mode */
|
|
M_WRTRAM(cpustate, (cpustate->memaccess+1), cpustate->ALU.w.l);
|
|
}
|
|
cpustate->Treg = cpustate->ALU.w.l;
|
|
cpustate->Preg.d = ( (INT16)cpustate->ALU.w.l * (INT16)M_RDROM(cpustate, cpustate->PFC) );
|
|
cpustate->PFC++;
|
|
cpustate->tms32025_dec_cycles += (2*CLK);
|
|
}
|
|
static void mar(tms32025_state *cpustate) /* LARP and NOP are a subset of this instruction */
|
|
{
|
|
if (cpustate->opcode.b.l & 0x80) MODIFY_AR_ARP(cpustate);
|
|
}
|
|
static void mpy(tms32025_state *cpustate)
|
|
{
|
|
GETDATA(cpustate, 0, 0);
|
|
cpustate->Preg.d = (INT16)(cpustate->ALU.w.l) * (INT16)(cpustate->Treg);
|
|
}
|
|
static void mpya(tms32025_state *cpustate)
|
|
{
|
|
cpustate->oldacc.d = cpustate->ACC.d;
|
|
SHIFT_Preg_TO_ALU(cpustate);
|
|
cpustate->ACC.d += cpustate->ALU.d;
|
|
CALCULATE_ADD_OVERFLOW(cpustate, cpustate->ALU.d);
|
|
CALCULATE_ADD_CARRY(cpustate);
|
|
GETDATA(cpustate, 0, 0);
|
|
cpustate->Preg.d = (INT16)(cpustate->ALU.w.l) * (INT16)(cpustate->Treg);
|
|
}
|
|
static void mpyk(tms32025_state *cpustate)
|
|
{
|
|
cpustate->Preg.d = (INT16)cpustate->Treg * ((INT16)(cpustate->opcode.w.l << 3) >> 3);
|
|
|
|
}
|
|
static void mpys(tms32025_state *cpustate)
|
|
{
|
|
cpustate->oldacc.d = cpustate->ACC.d;
|
|
SHIFT_Preg_TO_ALU(cpustate);
|
|
cpustate->ACC.d -= cpustate->ALU.d;
|
|
CALCULATE_SUB_OVERFLOW(cpustate, cpustate->ALU.d);
|
|
CALCULATE_SUB_CARRY(cpustate);
|
|
GETDATA(cpustate, 0, 0);
|
|
cpustate->Preg.d = (INT16)(cpustate->ALU.w.l) * (INT16)(cpustate->Treg);
|
|
}
|
|
static void mpyu(tms32025_state *cpustate)
|
|
{
|
|
GETDATA(cpustate, 0, 0);
|
|
cpustate->Preg.d = (UINT16)(cpustate->ALU.w.l) * (UINT16)(cpustate->Treg);
|
|
}
|
|
static void neg(tms32025_state *cpustate)
|
|
{
|
|
if (cpustate->ACC.d == 0x80000000) {
|
|
SET0(cpustate, OV_FLAG);
|
|
if (OVM) cpustate->ACC.d = 0x7fffffff;
|
|
}
|
|
else cpustate->ACC.d = -cpustate->ACC.d;
|
|
if (cpustate->ACC.d) CLR0(cpustate, C_FLAG);
|
|
else SET0(cpustate, C_FLAG);
|
|
}
|
|
/*
|
|
static void nop(tms32025_state *cpustate) { } // NOP is a subset of the MAR instruction
|
|
*/
|
|
static void norm(tms32025_state *cpustate)
|
|
{
|
|
UINT32 acc = cpustate->ACC.d;
|
|
|
|
if( acc == 0 || ((acc^(acc<<1))&(1<<31))!=0 ) {
|
|
SET1(cpustate, TC_FLAG); /* 1 -> TC */
|
|
}
|
|
else {
|
|
CLR1(cpustate, TC_FLAG); /* 0 -> TC */
|
|
cpustate->ACC.d <<= 1; /* (ACC)*2 -> ACC */
|
|
MODIFY_AR_ARP(cpustate);
|
|
}
|
|
}
|
|
static void or_(tms32025_state *cpustate)
|
|
{
|
|
GETDATA(cpustate, 0, 0);
|
|
cpustate->ACC.w.l |= cpustate->ALU.w.l;
|
|
}
|
|
static void ork(tms32025_state *cpustate)
|
|
{
|
|
cpustate->ALU.d = (UINT16)M_RDOP_ARG(cpustate->PC);
|
|
cpustate->PC++;
|
|
cpustate->ALU.d <<= (cpustate->opcode.b.h & 0xf);
|
|
cpustate->ACC.d |= (cpustate->ALU.d & 0x7fffffff);
|
|
}
|
|
static void out(tms32025_state *cpustate)
|
|
{
|
|
GETDATA(cpustate, 0, 0);
|
|
P_OUT( (cpustate->opcode.b.h & 0xf), cpustate->ALU.w.l );
|
|
}
|
|
static void pac(tms32025_state *cpustate)
|
|
{
|
|
SHIFT_Preg_TO_ALU(cpustate);
|
|
cpustate->ACC.d = cpustate->ALU.d;
|
|
}
|
|
static void pop(tms32025_state *cpustate)
|
|
{
|
|
cpustate->ACC.d = (UINT16)POP_STACK(cpustate);
|
|
}
|
|
static void popd(tms32025_state *cpustate)
|
|
{
|
|
cpustate->ALU.d = (UINT16)POP_STACK(cpustate);
|
|
PUTDATA(cpustate, cpustate->ALU.w.l);
|
|
}
|
|
static void pshd(tms32025_state *cpustate)
|
|
{
|
|
GETDATA(cpustate, 0, 0);
|
|
PUSH_STACK(cpustate, cpustate->ALU.w.l);
|
|
}
|
|
static void push(tms32025_state *cpustate)
|
|
{
|
|
PUSH_STACK(cpustate, cpustate->ACC.w.l);
|
|
}
|
|
static void rc(tms32025_state *cpustate)
|
|
{
|
|
CLR1(cpustate, C_FLAG);
|
|
}
|
|
static void ret(tms32025_state *cpustate)
|
|
{
|
|
SET_PC(POP_STACK(cpustate));
|
|
}
|
|
static void rfsm(tms32025_state *cpustate) /** serial port mode */
|
|
{
|
|
CLR1(cpustate, FSM_FLAG);
|
|
}
|
|
static void rhm(tms32025_state *cpustate)
|
|
{
|
|
CLR1(cpustate, HM_FLAG);
|
|
}
|
|
static void rol(tms32025_state *cpustate)
|
|
{
|
|
cpustate->ALU.d = cpustate->ACC.d;
|
|
cpustate->ACC.d <<= 1;
|
|
if (CARRY) cpustate->ACC.d |= 1;
|
|
if (cpustate->ALU.d & 0x80000000) SET1(cpustate, C_FLAG);
|
|
else CLR1(cpustate, C_FLAG);
|
|
}
|
|
static void ror(tms32025_state *cpustate)
|
|
{
|
|
cpustate->ALU.d = cpustate->ACC.d;
|
|
cpustate->ACC.d >>= 1;
|
|
if (CARRY) cpustate->ACC.d |= 0x80000000;
|
|
if (cpustate->ALU.d & 1) SET1(cpustate, C_FLAG);
|
|
else CLR1(cpustate, C_FLAG);
|
|
}
|
|
static void rovm(tms32025_state *cpustate)
|
|
{
|
|
CLR0(cpustate, OVM_FLAG);
|
|
}
|
|
static void rpt(tms32025_state *cpustate)
|
|
{
|
|
GETDATA(cpustate, 0, 0);
|
|
cpustate->RPTC = cpustate->ALU.b.l;
|
|
cpustate->init_load_addr = 2; /* Initiate repeat mode */
|
|
}
|
|
static void rptk(tms32025_state *cpustate)
|
|
{
|
|
cpustate->RPTC = cpustate->opcode.b.l;
|
|
cpustate->init_load_addr = 2; /* Initiate repeat mode */
|
|
}
|
|
static void rsxm(tms32025_state *cpustate)
|
|
{
|
|
CLR1(cpustate, SXM_FLAG);
|
|
}
|
|
static void rtc(tms32025_state *cpustate)
|
|
{
|
|
CLR1(cpustate, TC_FLAG);
|
|
}
|
|
static void rtxm(tms32025_state *cpustate) /** Serial port stuff */
|
|
{
|
|
CLR1(cpustate, TXM_FLAG);
|
|
}
|
|
static void rxf(tms32025_state *cpustate)
|
|
{
|
|
CLR1(cpustate, XF_FLAG);
|
|
S_OUT(TMS32025_XF,CLEAR_LINE);
|
|
}
|
|
static void sach(tms32025_state *cpustate)
|
|
{
|
|
cpustate->ALU.d = (cpustate->ACC.d << (cpustate->opcode.b.h & 7));
|
|
PUTDATA(cpustate, cpustate->ALU.w.h);
|
|
}
|
|
static void sacl(tms32025_state *cpustate)
|
|
{
|
|
cpustate->ALU.d = (cpustate->ACC.d << (cpustate->opcode.b.h & 7));
|
|
PUTDATA(cpustate, cpustate->ALU.w.l);
|
|
}
|
|
static void sar_ar0(tms32025_state *cpustate) { PUTDATA(cpustate, cpustate->AR[0]); }
|
|
static void sar_ar1(tms32025_state *cpustate) { PUTDATA(cpustate, cpustate->AR[1]); }
|
|
static void sar_ar2(tms32025_state *cpustate) { PUTDATA(cpustate, cpustate->AR[2]); }
|
|
static void sar_ar3(tms32025_state *cpustate) { PUTDATA(cpustate, cpustate->AR[3]); }
|
|
static void sar_ar4(tms32025_state *cpustate) { PUTDATA(cpustate, cpustate->AR[4]); }
|
|
static void sar_ar5(tms32025_state *cpustate) { PUTDATA(cpustate, cpustate->AR[5]); }
|
|
static void sar_ar6(tms32025_state *cpustate) { PUTDATA(cpustate, cpustate->AR[6]); }
|
|
static void sar_ar7(tms32025_state *cpustate) { PUTDATA(cpustate, cpustate->AR[7]); }
|
|
|
|
static void sblk(tms32025_state *cpustate)
|
|
{
|
|
cpustate->oldacc.d = cpustate->ACC.d;
|
|
if (SXM) cpustate->ALU.d = (INT16)M_RDOP_ARG(cpustate->PC);
|
|
else cpustate->ALU.d = (UINT16)M_RDOP_ARG(cpustate->PC);
|
|
cpustate->PC++;
|
|
cpustate->ALU.d <<= (cpustate->opcode.b.h & 0xf);
|
|
cpustate->ACC.d -= cpustate->ALU.d;
|
|
CALCULATE_SUB_OVERFLOW(cpustate, cpustate->ALU.d);
|
|
CALCULATE_SUB_CARRY(cpustate);
|
|
}
|
|
static void sbrk_tms(tms32025_state *cpustate)
|
|
{
|
|
cpustate->AR[ARP] -= cpustate->opcode.b.l;
|
|
}
|
|
static void sc(tms32025_state *cpustate)
|
|
{
|
|
SET1(cpustate, C_FLAG);
|
|
}
|
|
static void sfl(tms32025_state *cpustate)
|
|
{
|
|
cpustate->ALU.d = cpustate->ACC.d;
|
|
cpustate->ACC.d <<= 1;
|
|
if (cpustate->ALU.d & 0x80000000) SET1(cpustate, C_FLAG);
|
|
else CLR1(cpustate, C_FLAG);
|
|
}
|
|
static void sfr(tms32025_state *cpustate)
|
|
{
|
|
cpustate->ALU.d = cpustate->ACC.d;
|
|
cpustate->ACC.d >>= 1;
|
|
if (SXM) {
|
|
if (cpustate->ALU.d & 0x80000000) cpustate->ACC.d |= 0x80000000;
|
|
}
|
|
if (cpustate->ALU.d & 1) SET1(cpustate, C_FLAG);
|
|
else CLR1(cpustate, C_FLAG);
|
|
}
|
|
static void sfsm(tms32025_state *cpustate) /** Serial port mode */
|
|
{
|
|
SET1(cpustate, FSM_FLAG);
|
|
}
|
|
static void shm(tms32025_state *cpustate)
|
|
{
|
|
SET1(cpustate, HM_FLAG);
|
|
}
|
|
static void sovm(tms32025_state *cpustate)
|
|
{
|
|
SET0(cpustate, OVM_FLAG);
|
|
}
|
|
static void spac(tms32025_state *cpustate)
|
|
{
|
|
cpustate->oldacc.d = cpustate->ACC.d;
|
|
SHIFT_Preg_TO_ALU(cpustate);
|
|
cpustate->ACC.d -= cpustate->ALU.d;
|
|
CALCULATE_SUB_OVERFLOW(cpustate, cpustate->ALU.d);
|
|
CALCULATE_SUB_CARRY(cpustate);
|
|
}
|
|
static void sph(tms32025_state *cpustate)
|
|
{
|
|
SHIFT_Preg_TO_ALU(cpustate);
|
|
PUTDATA(cpustate, cpustate->ALU.w.h);
|
|
}
|
|
static void spl(tms32025_state *cpustate)
|
|
{
|
|
SHIFT_Preg_TO_ALU(cpustate);
|
|
PUTDATA(cpustate, cpustate->ALU.w.l);
|
|
}
|
|
static void spm(tms32025_state *cpustate)
|
|
{
|
|
MODIFY_PM(cpustate, (cpustate->opcode.b.l & 3) );
|
|
}
|
|
static void sqra(tms32025_state *cpustate)
|
|
{
|
|
cpustate->oldacc.d = cpustate->ACC.d;
|
|
SHIFT_Preg_TO_ALU(cpustate);
|
|
cpustate->ACC.d += cpustate->ALU.d;
|
|
CALCULATE_ADD_OVERFLOW(cpustate, cpustate->ALU.d);
|
|
CALCULATE_ADD_CARRY(cpustate);
|
|
GETDATA(cpustate, 0, 0);
|
|
cpustate->Treg = cpustate->ALU.w.l;
|
|
cpustate->Preg.d = ((INT16)cpustate->ALU.w.l * (INT16)cpustate->ALU.w.l);
|
|
}
|
|
static void sqrs(tms32025_state *cpustate)
|
|
{
|
|
cpustate->oldacc.d = cpustate->ACC.d;
|
|
SHIFT_Preg_TO_ALU(cpustate);
|
|
cpustate->ACC.d -= cpustate->ALU.d;
|
|
CALCULATE_SUB_OVERFLOW(cpustate, cpustate->ALU.d);
|
|
CALCULATE_SUB_CARRY(cpustate);
|
|
GETDATA(cpustate, 0, 0);
|
|
cpustate->Treg = cpustate->ALU.w.l;
|
|
cpustate->Preg.d = ((INT16)cpustate->ALU.w.l * (INT16)cpustate->ALU.w.l);
|
|
}
|
|
static void sst(tms32025_state *cpustate)
|
|
{
|
|
PUTDATA_SST(cpustate, cpustate->STR0);
|
|
}
|
|
static void sst1(tms32025_state *cpustate)
|
|
{
|
|
PUTDATA_SST(cpustate, cpustate->STR1);
|
|
}
|
|
static void ssxm(tms32025_state *cpustate)
|
|
{ /** Check instruction description, and make sure right instructions use SXM */
|
|
SET1(cpustate, SXM_FLAG);
|
|
}
|
|
static void stc(tms32025_state *cpustate)
|
|
{
|
|
SET1(cpustate, TC_FLAG);
|
|
}
|
|
static void stxm(tms32025_state *cpustate) /** Serial port stuff */
|
|
{
|
|
SET1(cpustate, TXM_FLAG);
|
|
}
|
|
static void sub(tms32025_state *cpustate)
|
|
{
|
|
cpustate->oldacc.d = cpustate->ACC.d;
|
|
GETDATA(cpustate, (cpustate->opcode.b.h & 0xf), SXM);
|
|
cpustate->ACC.d -= cpustate->ALU.d;
|
|
CALCULATE_SUB_OVERFLOW(cpustate, cpustate->ALU.d);
|
|
CALCULATE_SUB_CARRY(cpustate);
|
|
}
|
|
static void subb(tms32025_state *cpustate)
|
|
{
|
|
cpustate->oldacc.d = cpustate->ACC.d;
|
|
GETDATA(cpustate, 0, 0);
|
|
if (CARRY == 0) cpustate->ACC.d--;
|
|
cpustate->ACC.d -= cpustate->ALU.d;
|
|
CALCULATE_SUB_OVERFLOW(cpustate, cpustate->ALU.d);
|
|
if (cpustate->ACC.d == cpustate->oldacc.d) {} /* edge case, carry remains same */
|
|
else CALCULATE_SUB_CARRY(cpustate);
|
|
}
|
|
static void subc(tms32025_state *cpustate)
|
|
{
|
|
PAIR temp_alu;
|
|
cpustate->oldacc.d = cpustate->ACC.d;
|
|
GETDATA(cpustate, 15, SXM);
|
|
cpustate->ACC.d -= cpustate->ALU.d; /* Temporary switch to ACC. Actual calculation is done as (ACC)-[mem] -> ALU, will be preserved later on. */
|
|
temp_alu.d = cpustate->ACC.d;
|
|
if ((INT32)((cpustate->oldacc.d ^ cpustate->ALU.d) & (cpustate->oldacc.d ^ cpustate->ACC.d)) < 0) {
|
|
SET0(cpustate, OV_FLAG); /* Not affected by OVM */
|
|
}
|
|
CALCULATE_SUB_CARRY(cpustate);
|
|
if( cpustate->oldacc.d >= cpustate->ALU.d ) {
|
|
cpustate->ACC.d = (cpustate->oldacc.d - cpustate->ALU.d)*2+1;
|
|
}
|
|
else {
|
|
cpustate->ACC.d = cpustate->oldacc.d*2;
|
|
}
|
|
cpustate->ALU.d = temp_alu.d;
|
|
}
|
|
static void subh(tms32025_state *cpustate)
|
|
{
|
|
cpustate->oldacc.d = cpustate->ACC.d;
|
|
GETDATA(cpustate, 0, 0);
|
|
cpustate->ACC.w.h -= cpustate->ALU.w.l;
|
|
if ((INT16)((cpustate->oldacc.w.h ^ cpustate->ALU.w.l) & (cpustate->oldacc.w.h ^ cpustate->ACC.w.h)) < 0) {
|
|
SET0(cpustate, OV_FLAG);
|
|
if (OVM)
|
|
cpustate->ACC.w.h = ((INT16)cpustate->oldacc.w.h < 0) ? 0x8000 : 0x7fff;
|
|
}
|
|
if ( ((INT16)(cpustate->oldacc.w.h) >= 0) && ((INT16)(cpustate->ACC.w.h) < 0) ) {
|
|
CLR1(cpustate, C_FLAG);
|
|
}
|
|
/* Carry flag is not affected, if no borrow occurred */
|
|
}
|
|
static void subk(tms32025_state *cpustate)
|
|
{
|
|
cpustate->oldacc.d = cpustate->ACC.d;
|
|
cpustate->ALU.d = (UINT8)cpustate->opcode.b.l;
|
|
cpustate->ACC.d -= cpustate->ALU.b.l;
|
|
CALCULATE_SUB_OVERFLOW(cpustate, cpustate->ALU.d);
|
|
CALCULATE_SUB_CARRY(cpustate);
|
|
}
|
|
static void subs(tms32025_state *cpustate)
|
|
{
|
|
cpustate->oldacc.d = cpustate->ACC.d;
|
|
GETDATA(cpustate, 0, 0);
|
|
cpustate->ACC.d -= cpustate->ALU.w.l;
|
|
CALCULATE_SUB_OVERFLOW(cpustate, cpustate->ALU.d);
|
|
CALCULATE_SUB_CARRY(cpustate);
|
|
}
|
|
static void subt(tms32025_state *cpustate)
|
|
{
|
|
cpustate->oldacc.d = cpustate->ACC.d;
|
|
GETDATA(cpustate, (cpustate->Treg & 0xf), SXM);
|
|
cpustate->ACC.d -= cpustate->ALU.d;
|
|
CALCULATE_SUB_OVERFLOW(cpustate, cpustate->ALU.d);
|
|
CALCULATE_SUB_CARRY(cpustate);
|
|
}
|
|
static void sxf(tms32025_state *cpustate)
|
|
{
|
|
SET1(cpustate, XF_FLAG);
|
|
S_OUT(TMS32025_XF,ASSERT_LINE);
|
|
}
|
|
static void tblr(tms32025_state *cpustate)
|
|
{
|
|
if (cpustate->init_load_addr) {
|
|
cpustate->PFC = cpustate->ACC.w.l;
|
|
}
|
|
cpustate->ALU.w.l = M_RDROM(cpustate, cpustate->PFC);
|
|
if ( (CNF0) && ( (UINT16)(cpustate->PFC) >= 0xff00 ) ) {} /** TMS32025 only */
|
|
else cpustate->tms32025_dec_cycles += (1*CLK);
|
|
PUTDATA(cpustate, cpustate->ALU.w.l);
|
|
cpustate->PFC++;
|
|
}
|
|
static void tblw(tms32025_state *cpustate)
|
|
{
|
|
if (cpustate->init_load_addr) {
|
|
cpustate->PFC = cpustate->ACC.w.l;
|
|
}
|
|
cpustate->tms32025_dec_cycles += (1*CLK);
|
|
GETDATA(cpustate, 0, 0);
|
|
if (cpustate->external_mem_access) cpustate->tms32025_dec_cycles += (1*CLK);
|
|
M_WRTROM(cpustate, cpustate->PFC, cpustate->ALU.w.l);
|
|
cpustate->PFC++;
|
|
}
|
|
static void trap(tms32025_state *cpustate)
|
|
{
|
|
PUSH_STACK(cpustate, cpustate->PC);
|
|
SET_PC(0x001E); /* Trap vector */
|
|
}
|
|
static void xor_(tms32025_state *cpustate)
|
|
{
|
|
GETDATA(cpustate, 0, 0);
|
|
cpustate->ACC.w.l ^= cpustate->ALU.w.l;
|
|
}
|
|
static void xork(tms32025_state *cpustate)
|
|
{
|
|
cpustate->oldacc.d = cpustate->ACC.d;
|
|
cpustate->ALU.d = M_RDOP_ARG(cpustate->PC);
|
|
cpustate->PC++;
|
|
cpustate->ALU.d <<= (cpustate->opcode.b.h & 0xf);
|
|
cpustate->ACC.d ^= cpustate->ALU.d;
|
|
cpustate->ACC.d |= (cpustate->oldacc.d & 0x80000000);
|
|
}
|
|
static void zalh(tms32025_state *cpustate)
|
|
{
|
|
GETDATA(cpustate, 0, 0);
|
|
cpustate->ACC.w.h = cpustate->ALU.w.l;
|
|
cpustate->ACC.w.l = 0x0000;
|
|
}
|
|
static void zalr(tms32025_state *cpustate)
|
|
{
|
|
GETDATA(cpustate, 0, 0);
|
|
cpustate->ACC.w.h = cpustate->ALU.w.l;
|
|
cpustate->ACC.w.l = 0x8000;
|
|
}
|
|
static void zals(tms32025_state *cpustate)
|
|
{
|
|
GETDATA(cpustate, 0, 0);
|
|
cpustate->ACC.w.l = cpustate->ALU.w.l;
|
|
cpustate->ACC.w.h = 0x0000;
|
|
}
|
|
|
|
|
|
/***********************************************************************
|
|
* Opcode Table (Cycles, Instruction)
|
|
***********************************************************************/
|
|
|
|
static const tms32025_opcode opcode_main[256]=
|
|
{
|
|
/*00*/ {1*CLK, add },{1*CLK, add },{1*CLK, add },{1*CLK, add },{1*CLK, add },{1*CLK, add },{1*CLK, add },{1*CLK, add },
|
|
/*08*/ {1*CLK, add },{1*CLK, add },{1*CLK, add },{1*CLK, add },{1*CLK, add },{1*CLK, add },{1*CLK, add },{1*CLK, add },
|
|
/*10*/ {1*CLK, sub },{1*CLK, sub },{1*CLK, sub },{1*CLK, sub },{1*CLK, sub },{1*CLK, sub },{1*CLK, sub },{1*CLK, sub },
|
|
/*18*/ {1*CLK, sub },{1*CLK, sub },{1*CLK, sub },{1*CLK, sub },{1*CLK, sub },{1*CLK, sub },{1*CLK, sub },{1*CLK, sub },
|
|
/*20*/ {1*CLK, lac },{1*CLK, lac },{1*CLK, lac },{1*CLK, lac },{1*CLK, lac },{1*CLK, lac },{1*CLK, lac },{1*CLK, lac },
|
|
/*28*/ {1*CLK, lac },{1*CLK, lac },{1*CLK, lac },{1*CLK, lac },{1*CLK, lac },{1*CLK, lac },{1*CLK, lac },{1*CLK, lac },
|
|
/*30*/ {1*CLK, lar_ar0 },{1*CLK, lar_ar1 },{1*CLK, lar_ar2 },{1*CLK, lar_ar3 },{1*CLK, lar_ar4 },{1*CLK, lar_ar5 },{1*CLK, lar_ar6 },{1*CLK, lar_ar7 },
|
|
/*38*/ {1*CLK, mpy },{1*CLK, sqra },{1*CLK, mpya },{1*CLK, mpys },{1*CLK, lt },{1*CLK, lta },{1*CLK, ltp },{1*CLK, ltd },
|
|
/*40*/ {1*CLK, zalh },{1*CLK, zals },{1*CLK, lact },{1*CLK, addc },{1*CLK, subh },{1*CLK, subs },{1*CLK, subt },{1*CLK, subc },
|
|
/*48*/ {1*CLK, addh },{1*CLK, adds },{1*CLK, addt },{1*CLK, rpt },{1*CLK, xor_ },{1*CLK, or_ },{1*CLK, and_ },{1*CLK, subb },
|
|
/*50*/ {1*CLK, lst },{1*CLK, lst1 },{1*CLK, ldp },{1*CLK, lph },{1*CLK, pshd },{1*CLK, mar },{1*CLK, dmov },{1*CLK, bitt },
|
|
/*58*/ {3*CLK, tblr },{2*CLK, tblw },{1*CLK, sqrs },{1*CLK, lts },{2*CLK, macd },{2*CLK, mac },{2*CLK, bc },{2*CLK, bnc },
|
|
/*60*/ {1*CLK, sacl },{1*CLK, sacl },{1*CLK, sacl },{1*CLK, sacl },{1*CLK, sacl },{1*CLK, sacl },{1*CLK, sacl },{1*CLK, sacl },
|
|
/*68*/ {1*CLK, sach },{1*CLK, sach },{1*CLK, sach },{1*CLK, sach },{1*CLK, sach },{1*CLK, sach },{1*CLK, sach },{1*CLK, sach },
|
|
/*70*/ {1*CLK, sar_ar0 },{1*CLK, sar_ar1 },{1*CLK, sar_ar2 },{1*CLK, sar_ar3 },{1*CLK, sar_ar4 },{1*CLK, sar_ar5 },{1*CLK, sar_ar6 },{1*CLK, sar_ar7 },
|
|
/*78*/ {1*CLK, sst },{1*CLK, sst1 },{1*CLK, popd },{1*CLK, zalr },{1*CLK, spl },{1*CLK, sph },{1*CLK, adrk },{1*CLK, sbrk_tms },
|
|
/*80*/ {2*CLK, in },{2*CLK, in },{2*CLK, in },{2*CLK, in },{2*CLK, in },{2*CLK, in },{2*CLK, in },{2*CLK, in },
|
|
/*88*/ {2*CLK, in },{2*CLK, in },{2*CLK, in },{2*CLK, in },{2*CLK, in },{2*CLK, in },{2*CLK, in },{2*CLK, in },
|
|
/*90*/ {1*CLK, bit },{1*CLK, bit },{1*CLK, bit },{1*CLK, bit },{1*CLK, bit },{1*CLK, bit },{1*CLK, bit },{1*CLK, bit },
|
|
/*98*/ {1*CLK, bit },{1*CLK, bit },{1*CLK, bit },{1*CLK, bit },{1*CLK, bit },{1*CLK, bit },{1*CLK, bit },{1*CLK, bit },
|
|
/*A0*/ {1*CLK, mpyk },{1*CLK, mpyk },{1*CLK, mpyk },{1*CLK, mpyk },{1*CLK, mpyk },{1*CLK, mpyk },{1*CLK, mpyk },{1*CLK, mpyk },
|
|
/*A8*/ {1*CLK, mpyk },{1*CLK, mpyk },{1*CLK, mpyk },{1*CLK, mpyk },{1*CLK, mpyk },{1*CLK, mpyk },{1*CLK, mpyk },{1*CLK, mpyk },
|
|
/*B0*/ {1*CLK, mpyk },{1*CLK, mpyk },{1*CLK, mpyk },{1*CLK, mpyk },{1*CLK, mpyk },{1*CLK, mpyk },{1*CLK, mpyk },{1*CLK, mpyk },
|
|
/*B8*/ {1*CLK, mpyk },{1*CLK, mpyk },{1*CLK, mpyk },{1*CLK, mpyk },{1*CLK, mpyk },{1*CLK, mpyk },{1*CLK, mpyk },{1*CLK, mpyk },
|
|
/*C0*/ {1*CLK, lark_ar0 },{1*CLK, lark_ar1 },{1*CLK, lark_ar2 },{1*CLK, lark_ar3 },{1*CLK, lark_ar4 },{1*CLK, lark_ar5 },{1*CLK, lark_ar6 },{1*CLK, lark_ar7 },
|
|
/*C8*/ {1*CLK, ldpk },{1*CLK, ldpk },{1*CLK, lack },{1*CLK, rptk },{1*CLK, addk },{1*CLK, subk },{1*CLK, opcodes_CE},{1*CLK, mpyu },
|
|
/*D0*/ {1*CLK,opcodes_Dx},{1*CLK, opcodes_Dx},{1*CLK, opcodes_Dx},{1*CLK, opcodes_Dx},{1*CLK, opcodes_Dx},{1*CLK, opcodes_Dx},{1*CLK, opcodes_Dx},{0*CLK, opcodes_Dx},
|
|
/*D8*/ {1*CLK,opcodes_Dx},{1*CLK, opcodes_Dx},{1*CLK, opcodes_Dx},{1*CLK, opcodes_Dx},{1*CLK, opcodes_Dx},{1*CLK, opcodes_Dx},{1*CLK, opcodes_Dx},{1*CLK, opcodes_Dx},
|
|
/*E0*/ {2*CLK, out },{2*CLK, out },{2*CLK, out },{2*CLK, out },{2*CLK, out },{2*CLK, out },{2*CLK, out },{2*CLK, out },
|
|
/*E8*/ {2*CLK, out },{2*CLK, out },{2*CLK, out },{2*CLK, out },{2*CLK, out },{2*CLK, out },{2*CLK, out },{2*CLK, out },
|
|
/*F0*/ {2*CLK, bv },{2*CLK, bgz },{2*CLK, blez },{2*CLK, blz },{2*CLK, bgez },{2*CLK, bnz },{2*CLK, bz },{2*CLK, bnv },
|
|
/*F8*/ {2*CLK, bbz },{2*CLK, bbnz },{2*CLK, bioz },{2*CLK, banz },{2*CLK, blkp },{2*CLK, blkd },{2*CLK, call },{2*CLK, br }
|
|
};
|
|
|
|
static const tms32025_opcode_CE opcode_CE_subset[256]= /* Instructions living under the CExx opcode */
|
|
{
|
|
/*00*/ {1*CLK, eint },{1*CLK, dint },{1*CLK, rovm },{1*CLK, sovm },{1*CLK, cnfd },{1*CLK, cnfp },{1*CLK, rsxm },{1*CLK, ssxm },
|
|
/*08*/ {1*CLK, spm },{1*CLK, spm },{1*CLK, spm },{1*CLK, spm },{1*CLK, rxf },{1*CLK, sxf },{1*CLK, fort },{1*CLK, fort },
|
|
/*10*/ {0*CLK, illegal },{0*CLK, illegal },{0*CLK, illegal },{0*CLK, illegal },{1*CLK, pac },{1*CLK, apac },{1*CLK, spac },{0*CLK, illegal },
|
|
/*18*/ {1*CLK, sfl },{1*CLK, sfr },{0*CLK, illegal },{1*CLK, abst },{1*CLK, push },{1*CLK, pop },{2*CLK, trap },{3*CLK, idle },
|
|
/*20*/ {1*CLK, rtxm },{1*CLK, stxm },{0*CLK, illegal },{1*CLK, neg },{2*CLK, cala },{2*CLK, bacc },{2*CLK, ret },{1*CLK, cmpl },
|
|
/*28*/ {0*CLK, illegal },{0*CLK, illegal },{0*CLK, illegal },{0*CLK, illegal },{0*CLK, illegal },{0*CLK, illegal },{0*CLK, illegal },{0*CLK, illegal },
|
|
/*30*/ {1*CLK, rc },{1*CLK, sc },{1*CLK, rtc },{1*CLK, stc },{1*CLK, rol },{1*CLK, ror },{1*CLK, rfsm },{1*CLK, sfsm },
|
|
/*38*/ {1*CLK, rhm },{1*CLK, shm },{0*CLK, illegal },{0*CLK, illegal },{1*CLK, conf },{1*CLK, conf },{1*CLK, conf },{1*CLK, conf },
|
|
/*40*/ {0*CLK, illegal },{0*CLK, illegal },{0*CLK, illegal },{0*CLK, illegal },{0*CLK, illegal },{0*CLK, illegal },{0*CLK, illegal },{0*CLK, illegal },
|
|
/*48*/ {0*CLK, illegal },{0*CLK, illegal },{0*CLK, illegal },{0*CLK, illegal },{0*CLK, illegal },{0*CLK, illegal },{0*CLK, illegal },{0*CLK, illegal },
|
|
/*50*/ {1*CLK, cmpr },{1*CLK, cmpr },{1*CLK, cmpr },{1*CLK, cmpr },{0*CLK, illegal },{0*CLK, illegal },{0*CLK, illegal },{0*CLK, illegal },
|
|
/*58*/ {0*CLK, illegal },{0*CLK, illegal },{0*CLK, illegal },{0*CLK, illegal },{0*CLK, illegal },{0*CLK, illegal },{0*CLK, illegal },{0*CLK, illegal },
|
|
/*60*/ {0*CLK, illegal },{0*CLK, illegal },{0*CLK, illegal },{0*CLK, illegal },{0*CLK, illegal },{0*CLK, illegal },{0*CLK, illegal },{0*CLK, illegal },
|
|
/*68*/ {0*CLK, illegal },{0*CLK, illegal },{0*CLK, illegal },{0*CLK, illegal },{0*CLK, illegal },{0*CLK, illegal },{0*CLK, illegal },{0*CLK, illegal },
|
|
/*70*/ {0*CLK, illegal },{0*CLK, illegal },{0*CLK, illegal },{0*CLK, illegal },{0*CLK, illegal },{0*CLK, illegal },{0*CLK, illegal },{0*CLK, illegal },
|
|
/*78*/ {0*CLK, illegal },{0*CLK, illegal },{0*CLK, illegal },{0*CLK, illegal },{0*CLK, illegal },{0*CLK, illegal },{0*CLK, illegal },{0*CLK, illegal },
|
|
/*80*/ {0*CLK, illegal },{0*CLK, illegal },{1*CLK, norm },{0*CLK, illegal },{0*CLK, illegal },{0*CLK, illegal },{0*CLK, illegal },{0*CLK, illegal },
|
|
/*88*/ {0*CLK, illegal },{0*CLK, illegal },{0*CLK, illegal },{0*CLK, illegal },{0*CLK, illegal },{0*CLK, illegal },{0*CLK, illegal },{0*CLK, illegal },
|
|
/*90*/ {0*CLK, illegal },{0*CLK, illegal },{1*CLK, norm },{0*CLK, illegal },{0*CLK, illegal },{0*CLK, illegal },{0*CLK, illegal },{0*CLK, illegal },
|
|
/*98*/ {0*CLK, illegal },{0*CLK, illegal },{0*CLK, illegal },{0*CLK, illegal },{0*CLK, illegal },{0*CLK, illegal },{0*CLK, illegal },{0*CLK, illegal },
|
|
/*A0*/ {0*CLK, illegal },{0*CLK, illegal },{1*CLK, norm },{0*CLK, illegal },{0*CLK, illegal },{0*CLK, illegal },{0*CLK, illegal },{0*CLK, illegal },
|
|
/*A8*/ {0*CLK, illegal },{0*CLK, illegal },{0*CLK, illegal },{0*CLK, illegal },{0*CLK, illegal },{0*CLK, illegal },{0*CLK, illegal },{0*CLK, illegal },
|
|
/*B0*/ {0*CLK, illegal },{0*CLK, illegal },{1*CLK, norm },{0*CLK, illegal },{0*CLK, illegal },{0*CLK, illegal },{0*CLK, illegal },{0*CLK, illegal },
|
|
/*B8*/ {0*CLK, illegal },{0*CLK, illegal },{0*CLK, illegal },{0*CLK, illegal },{0*CLK, illegal },{0*CLK, illegal },{0*CLK, illegal },{0*CLK, illegal },
|
|
/*C0*/ {0*CLK, illegal },{0*CLK, illegal },{1*CLK, norm },{0*CLK, illegal },{0*CLK, illegal },{0*CLK, illegal },{0*CLK, illegal },{0*CLK, illegal },
|
|
/*C8*/ {0*CLK, illegal },{0*CLK, illegal },{0*CLK, illegal },{0*CLK, illegal },{0*CLK, illegal },{0*CLK, illegal },{0*CLK, illegal },{0*CLK, illegal },
|
|
/*D0*/ {0*CLK, illegal },{0*CLK, illegal },{1*CLK, norm },{0*CLK, illegal },{0*CLK, illegal },{0*CLK, illegal },{0*CLK, illegal },{0*CLK, illegal },
|
|
/*D8*/ {0*CLK, illegal },{0*CLK, illegal },{0*CLK, illegal },{0*CLK, illegal },{0*CLK, illegal },{0*CLK, illegal },{0*CLK, illegal },{0*CLK, illegal },
|
|
/*E0*/ {0*CLK, illegal },{0*CLK, illegal },{1*CLK, norm },{0*CLK, illegal },{0*CLK, illegal },{0*CLK, illegal },{0*CLK, illegal },{0*CLK, illegal },
|
|
/*E8*/ {0*CLK, illegal },{0*CLK, illegal },{0*CLK, illegal },{0*CLK, illegal },{0*CLK, illegal },{0*CLK, illegal },{0*CLK, illegal },{0*CLK, illegal },
|
|
/*F0*/ {0*CLK, illegal },{0*CLK, illegal },{1*CLK, norm },{0*CLK, illegal },{0*CLK, illegal },{0*CLK, illegal },{0*CLK, illegal },{0*CLK, illegal },
|
|
/*F8*/ {0*CLK, illegal },{0*CLK, illegal },{0*CLK, illegal },{0*CLK, illegal },{0*CLK, illegal },{0*CLK, illegal },{0*CLK, illegal },{0*CLK, illegal }
|
|
};
|
|
|
|
static const tms32025_opcode_Dx opcode_Dx_subset[8]= /* Instructions living under the Dxxx opcode */
|
|
{
|
|
/*00*/ {2*CLK, lrlk },{2*CLK, lalk },{2*CLK, adlk },{2*CLK, sblk },{2*CLK, andk },{2*CLK, ork },{2*CLK, xork },{0*CLK, illegal }
|
|
};
|
|
|
|
|
|
|
|
/****************************************************************************
|
|
* Inits CPU emulation
|
|
****************************************************************************/
|
|
static CPU_INIT( tms32025 )
|
|
{
|
|
tms32025_state *cpustate = get_safe_token(device);
|
|
|
|
cpustate->intRAM = auto_alloc_array(device->machine(), UINT16, 0x800);
|
|
cpustate->irq_callback = irqcallback;
|
|
cpustate->device = device;
|
|
cpustate->program = device->space(AS_PROGRAM);
|
|
cpustate->direct = &cpustate->program->direct();
|
|
cpustate->data = device->space(AS_DATA);
|
|
cpustate->io = device->space(AS_IO);
|
|
|
|
device->save_item(NAME(cpustate->PC));
|
|
device->save_item(NAME(cpustate->STR0));
|
|
device->save_item(NAME(cpustate->STR1));
|
|
device->save_item(NAME(cpustate->PFC));
|
|
device->save_item(NAME(cpustate->IFR));
|
|
device->save_item(NAME(cpustate->RPTC));
|
|
device->save_item(NAME(cpustate->ACC.d));
|
|
device->save_item(NAME(cpustate->ALU.d));
|
|
device->save_item(NAME(cpustate->Preg.d));
|
|
device->save_item(NAME(cpustate->Treg));
|
|
device->save_item(NAME(cpustate->AR[0]));
|
|
device->save_item(NAME(cpustate->AR[1]));
|
|
device->save_item(NAME(cpustate->AR[2]));
|
|
device->save_item(NAME(cpustate->AR[3]));
|
|
device->save_item(NAME(cpustate->AR[4]));
|
|
device->save_item(NAME(cpustate->AR[5]));
|
|
device->save_item(NAME(cpustate->AR[6]));
|
|
device->save_item(NAME(cpustate->AR[7]));
|
|
device->save_item(NAME(cpustate->STACK[0]));
|
|
device->save_item(NAME(cpustate->STACK[1]));
|
|
device->save_item(NAME(cpustate->STACK[2]));
|
|
device->save_item(NAME(cpustate->STACK[3]));
|
|
device->save_item(NAME(cpustate->STACK[4]));
|
|
device->save_item(NAME(cpustate->STACK[5]));
|
|
device->save_item(NAME(cpustate->STACK[6]));
|
|
device->save_item(NAME(cpustate->STACK[7]));
|
|
|
|
device->save_item(NAME(cpustate->oldacc));
|
|
device->save_item(NAME(cpustate->memaccess));
|
|
device->save_item(NAME(cpustate->icount));
|
|
device->save_item(NAME(cpustate->mHackIgnoreARP));
|
|
|
|
device->save_item(NAME(cpustate->idle));
|
|
device->save_item(NAME(cpustate->hold));
|
|
device->save_item(NAME(cpustate->external_mem_access));
|
|
device->save_item(NAME(cpustate->init_load_addr));
|
|
device->save_item(NAME(cpustate->PREVPC));
|
|
|
|
// device->save_pointer(NAME(cpustate->intRAM), 0x800*2);
|
|
}
|
|
|
|
/****************************************************************************
|
|
* Reset registers to their initial values
|
|
****************************************************************************/
|
|
static CPU_RESET( tms32025 )
|
|
{
|
|
tms32025_state *cpustate = get_safe_token(device);
|
|
|
|
SET_PC(0); /* Starting address on a reset */
|
|
cpustate->STR0 |= 0x0600; /* INTM and unused bit set to 1 */
|
|
cpustate->STR0 &= 0xefff; /* OV cleared to 0. Remaining bits undefined */
|
|
cpustate->STR1 |= 0x07f0; /* SXM, C, HM, FSM, XF and unused bits set to 1 */
|
|
cpustate->STR1 &= 0xeff0; /* CNF, FO, TXM, PM bits cleared to 0. Remaining bits undefined */
|
|
cpustate->RPTC = 0; /* Reset repeat counter to 0 */
|
|
cpustate->IFR = 0; /* IRQ pending flags */
|
|
|
|
S_OUT(TMS32025_XF,ASSERT_LINE); /* XF flag is high. Must set the pin */
|
|
|
|
/* Set the internal memory mapped registers */
|
|
GREG = 0;
|
|
TIM = 0xffff;
|
|
PRD = 0xffff;
|
|
IMR = 0xffc0;
|
|
|
|
cpustate->idle = 0;
|
|
cpustate->hold = 0;
|
|
cpustate->init_load_addr = 1;
|
|
|
|
/* Reset the Data/Program address banks */
|
|
memset(cpustate->pgmmap, 0, sizeof(cpustate->pgmmap));
|
|
memset(cpustate->datamap, 0, sizeof(cpustate->datamap));
|
|
|
|
cpustate->datamap[0] = &cpustate->intRAM[0x000]; /* B2 */
|
|
cpustate->datamap[4] = &cpustate->intRAM[0x200]; /* B0 */
|
|
cpustate->datamap[5] = &cpustate->intRAM[0x280]; /* B0 */
|
|
cpustate->datamap[6] = &cpustate->intRAM[0x300]; /* B1 */
|
|
cpustate->datamap[7] = &cpustate->intRAM[0x380]; /* B1 */
|
|
}
|
|
|
|
static CPU_RESET( tms32026 )
|
|
{
|
|
tms32025_state *cpustate = get_safe_token(device);
|
|
|
|
CPU_RESET_CALL(tms32025);
|
|
|
|
/* Reset the Data/Program address banks */
|
|
memset(cpustate->pgmmap, 0, sizeof(cpustate->pgmmap));
|
|
memset(cpustate->datamap, 0, sizeof(cpustate->datamap));
|
|
|
|
cpustate->datamap[0] = &cpustate->intRAM[0x000]; /* B2 */
|
|
cpustate->datamap[4] = &cpustate->intRAM[0x200]; /* B0 */
|
|
cpustate->datamap[5] = &cpustate->intRAM[0x280]; /* B0 */
|
|
cpustate->datamap[6] = &cpustate->intRAM[0x300]; /* B0 */
|
|
cpustate->datamap[7] = &cpustate->intRAM[0x380]; /* B0 */
|
|
cpustate->datamap[8] = &cpustate->intRAM[0x400]; /* B1 */
|
|
cpustate->datamap[9] = &cpustate->intRAM[0x480]; /* B1 */
|
|
cpustate->datamap[10] = &cpustate->intRAM[0x500]; /* B1 */
|
|
cpustate->datamap[11] = &cpustate->intRAM[0x580]; /* B1 */
|
|
cpustate->datamap[12] = &cpustate->intRAM[0x600]; /* B3 */
|
|
cpustate->datamap[13] = &cpustate->intRAM[0x680]; /* B3 */
|
|
cpustate->datamap[14] = &cpustate->intRAM[0x700]; /* B3 */
|
|
cpustate->datamap[15] = &cpustate->intRAM[0x780]; /* B3 */
|
|
}
|
|
|
|
|
|
/****************************************************************************
|
|
* Shut down CPU emulation
|
|
****************************************************************************/
|
|
static CPU_EXIT( tms32025 )
|
|
{
|
|
}
|
|
|
|
|
|
/****************************************************************************
|
|
* Issue an interrupt if necessary
|
|
****************************************************************************/
|
|
static int process_IRQs(tms32025_state *cpustate)
|
|
{
|
|
/********** Interrupt Flag Register (IFR) **********
|
|
| 5 | 4 | 3 | 2 | 1 | 0 |
|
|
| XINT| RINT| TINT| INT2| INT1| INT0|
|
|
*/
|
|
|
|
cpustate->tms32025_irq_cycles = 0;
|
|
|
|
/* Dont service Interrupts if masked, or prev instruction was EINT ! */
|
|
|
|
if ( (INTM == 0) && (cpustate->opcode.w.l != 0xce00) && (cpustate->IFR & IMR) )
|
|
{
|
|
cpustate->tms32025_irq_cycles = (3*CLK); /* 3 clock cycles used due to PUSH and DINT operation ? */
|
|
PUSH_STACK(cpustate, cpustate->PC);
|
|
|
|
if ((cpustate->IFR & 0x01) && (IMR & 0x01)) { /* IRQ line 0 */
|
|
//logerror("TMS32025: Active INT0\n");
|
|
SET_PC(0x0002);
|
|
(*cpustate->irq_callback)(cpustate->device, 0);
|
|
cpustate->idle = 0;
|
|
cpustate->IFR &= (~0x01);
|
|
SET0(cpustate, INTM_FLAG);
|
|
return cpustate->tms32025_irq_cycles;
|
|
}
|
|
if ((cpustate->IFR & 0x02) && (IMR & 0x02)) { /* IRQ line 1 */
|
|
//logerror("TMS32025: Active INT1\n");
|
|
SET_PC(0x0004);
|
|
(*cpustate->irq_callback)(cpustate->device, 1);
|
|
cpustate->idle = 0;
|
|
cpustate->IFR &= (~0x02);
|
|
SET0(cpustate, INTM_FLAG);
|
|
return cpustate->tms32025_irq_cycles;
|
|
}
|
|
if ((cpustate->IFR & 0x04) && (IMR & 0x04)) { /* IRQ line 2 */
|
|
//logerror("TMS32025: Active INT2\n");
|
|
SET_PC(0x0006);
|
|
(*cpustate->irq_callback)(cpustate->device, 2);
|
|
cpustate->idle = 0;
|
|
cpustate->IFR &= (~0x04);
|
|
SET0(cpustate, INTM_FLAG);
|
|
return cpustate->tms32025_irq_cycles;
|
|
}
|
|
if ((cpustate->IFR & 0x08) && (IMR & 0x08)) { /* Timer IRQ (internal) */
|
|
// logerror("TMS32025: Active TINT (Timer)\n");
|
|
SET_PC(0x0018);
|
|
cpustate->idle = 0;
|
|
cpustate->IFR &= (~0x08);
|
|
SET0(cpustate, INTM_FLAG);
|
|
return cpustate->tms32025_irq_cycles;
|
|
}
|
|
if ((cpustate->IFR & 0x10) && (IMR & 0x10)) { /* Serial port receive IRQ (internal) */
|
|
// logerror("TMS32025: Active RINT (Serial receive)\n");
|
|
DRR = S_IN(TMS32025_DR);
|
|
SET_PC(0x001A);
|
|
cpustate->idle = 0;
|
|
cpustate->IFR &= (~0x10);
|
|
SET0(cpustate, INTM_FLAG);
|
|
return cpustate->tms32025_irq_cycles;
|
|
}
|
|
if ((cpustate->IFR & 0x20) && (IMR & 0x20)) { /* Serial port transmit IRQ (internal) */
|
|
// logerror("TMS32025: Active XINT (Serial transmit)\n");
|
|
S_OUT(TMS32025_DX,DXR);
|
|
SET_PC(0x001C);
|
|
cpustate->idle = 0;
|
|
cpustate->IFR &= (~0x20);
|
|
SET0(cpustate, INTM_FLAG);
|
|
return cpustate->tms32025_irq_cycles;
|
|
}
|
|
}
|
|
return cpustate->tms32025_irq_cycles;
|
|
}
|
|
|
|
static void set_fsx_line(tms32025_state *cpustate, int state)
|
|
{
|
|
if (state != CLEAR_LINE && cpustate->waiting_for_serial_frame)
|
|
{
|
|
cpustate->waiting_for_serial_frame = 0;
|
|
cpustate->IFR = 0x20;
|
|
}
|
|
}
|
|
|
|
INLINE void process_timer(tms32025_state *cpustate, int clocks)
|
|
{
|
|
int preclocks, ticks;
|
|
|
|
/* easy case: no actual ticks */
|
|
again:
|
|
preclocks = CLK - cpustate->timerover;
|
|
if (clocks < preclocks)
|
|
{
|
|
cpustate->timerover += clocks;
|
|
cpustate->icount -= clocks;
|
|
return;
|
|
}
|
|
|
|
/* if we're not going to overflow the timer, just count the clocks */
|
|
ticks = 1 + (clocks - preclocks) / CLK;
|
|
if (ticks <= TIM)
|
|
{
|
|
cpustate->icount -= clocks;
|
|
cpustate->timerover = clocks - (ticks - 1) * CLK - preclocks;
|
|
TIM -= ticks;
|
|
}
|
|
|
|
/* otherwise, overflow the timer and signal an interrupt */
|
|
else
|
|
{
|
|
cpustate->icount -= preclocks + CLK * TIM;
|
|
cpustate->timerover = 0;
|
|
TIM = PRD;
|
|
|
|
cpustate->IFR |= 0x08;
|
|
clocks = process_IRQs(cpustate); /* Handle Timer IRQ */
|
|
goto again;
|
|
}
|
|
}
|
|
|
|
|
|
/****************************************************************************
|
|
* Execute ICount cycles. Exit when 0 or less
|
|
****************************************************************************/
|
|
static CPU_EXECUTE( tms32025 )
|
|
{
|
|
tms32025_state *cpustate = get_safe_token(device);
|
|
|
|
|
|
/**** Respond to external hold signal */
|
|
if (S_IN(TMS32025_HOLD) == ASSERT_LINE) {
|
|
if (cpustate->hold == 0) {
|
|
S_OUT(TMS32025_HOLDA,ASSERT_LINE); /* Hold-Ack (active low) */
|
|
}
|
|
cpustate->hold = 1;
|
|
if (HM) {
|
|
cpustate->icount = 0; /* Exit */
|
|
}
|
|
else {
|
|
if (cpustate->external_mem_access) {
|
|
cpustate->icount = 0; /* Exit */
|
|
}
|
|
}
|
|
}
|
|
else {
|
|
if (cpustate->hold == 1) {
|
|
S_OUT(TMS32025_HOLDA,CLEAR_LINE); /* Hold-Ack (active low) */
|
|
process_timer(cpustate, 3);
|
|
}
|
|
cpustate->hold = 0;
|
|
}
|
|
|
|
/**** If idling, update timer and/or exit execution, but test for irqs first */
|
|
if (cpustate->idle && cpustate->IFR && cpustate->icount > 0)
|
|
cpustate->icount -= process_IRQs(cpustate);
|
|
|
|
while (cpustate->idle && cpustate->icount > 0)
|
|
process_timer(cpustate, cpustate->icount);
|
|
|
|
if (cpustate->icount <= 0) debugger_instruction_hook(device, cpustate->PC);
|
|
|
|
|
|
while (cpustate->icount > 0)
|
|
{
|
|
cpustate->tms32025_dec_cycles = 0;
|
|
|
|
if (cpustate->IFR) { /* Check IRQ Flag Register for pending IRQs */
|
|
cpustate->tms32025_dec_cycles += process_IRQs(cpustate);
|
|
}
|
|
|
|
cpustate->PREVPC = cpustate->PC;
|
|
|
|
debugger_instruction_hook(device, cpustate->PC);
|
|
|
|
cpustate->opcode.d = M_RDOP(cpustate->PC);
|
|
cpustate->PC++;
|
|
|
|
if (cpustate->opcode.b.h == 0xCE) /* Opcode 0xCExx has many sub-opcodes in its minor byte */
|
|
{
|
|
cpustate->tms32025_dec_cycles += opcode_CE_subset[cpustate->opcode.b.l].cycles;
|
|
(*opcode_CE_subset[cpustate->opcode.b.l].function)(cpustate);
|
|
}
|
|
else if ((cpustate->opcode.w.l & 0xf0f8) == 0xd000) /* Opcode 0xDxxx has many sub-opcodes in its minor byte */
|
|
{
|
|
cpustate->tms32025_dec_cycles += opcode_Dx_subset[cpustate->opcode.b.l].cycles;
|
|
(*opcode_Dx_subset[cpustate->opcode.b.l].function)(cpustate);
|
|
}
|
|
else /* Do all opcodes except the CExx and Dxxx ones */
|
|
{
|
|
cpustate->tms32025_dec_cycles += opcode_main[cpustate->opcode.b.h].cycles;
|
|
(*opcode_main[cpustate->opcode.b.h].function)(cpustate);
|
|
}
|
|
|
|
|
|
if (cpustate->init_load_addr == 2) { /* Repeat next instruction */
|
|
cpustate->PREVPC = cpustate->PC;
|
|
|
|
debugger_instruction_hook(device, cpustate->PC);
|
|
|
|
cpustate->opcode.d = M_RDOP(cpustate->PC);
|
|
cpustate->PC++;
|
|
cpustate->tms32025_dec_cycles += (1*CLK);
|
|
|
|
do {
|
|
if (cpustate->opcode.b.h == 0xCE)
|
|
{ /* Do all 0xCExx Opcodes */
|
|
if (cpustate->init_load_addr) {
|
|
cpustate->tms32025_dec_cycles += (1*CLK);
|
|
}
|
|
else {
|
|
cpustate->tms32025_dec_cycles += (1*CLK);
|
|
}
|
|
(*opcode_CE_subset[cpustate->opcode.b.l].function)(cpustate);
|
|
}
|
|
if ((cpustate->opcode.w.l & 0xf0f8) == 0xd000)
|
|
{ /* Do all valid 0xDxxx Opcodes */
|
|
if (cpustate->init_load_addr) {
|
|
cpustate->tms32025_dec_cycles += (1*CLK);
|
|
}
|
|
else {
|
|
cpustate->tms32025_dec_cycles += (1*CLK);
|
|
}
|
|
(*opcode_Dx_subset[cpustate->opcode.b.l].function)(cpustate);
|
|
}
|
|
else
|
|
{ /* Do all other opcodes */
|
|
if (cpustate->init_load_addr) {
|
|
cpustate->tms32025_dec_cycles += (1*CLK);
|
|
}
|
|
else {
|
|
cpustate->tms32025_dec_cycles += (1*CLK);
|
|
}
|
|
(*opcode_main[cpustate->opcode.b.h].function)(cpustate);
|
|
}
|
|
cpustate->init_load_addr = 0;
|
|
cpustate->RPTC-- ;
|
|
} while ((INT8)(cpustate->RPTC) != -1);
|
|
cpustate->RPTC = 0;
|
|
cpustate->PFC = cpustate->PC;
|
|
cpustate->init_load_addr = 1;
|
|
}
|
|
|
|
process_timer(cpustate, cpustate->tms32025_dec_cycles);
|
|
|
|
/**** If device is put into idle mode, exit and wait for an interrupt */
|
|
while (cpustate->idle && cpustate->icount > 0)
|
|
process_timer(cpustate, cpustate->icount);
|
|
|
|
|
|
/**** If hold pin is active, exit if accessing external memory or if HM is set */
|
|
if (cpustate->hold) {
|
|
if (cpustate->external_mem_access || (HM)) {
|
|
if (cpustate->icount > 0) {
|
|
cpustate->icount = 0;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
|
|
/****************************************************************************
|
|
* Set IRQ line state
|
|
****************************************************************************/
|
|
static void set_irq_line(tms32025_state *cpustate, int irqline, int state)
|
|
{
|
|
/* Pending IRQs cannot be cleared */
|
|
|
|
if (state != CLEAR_LINE)
|
|
{
|
|
cpustate->IFR |= (1 << irqline);
|
|
// cpustate->IFR &= 0x07;
|
|
}
|
|
}
|
|
|
|
|
|
/****************************************************************************
|
|
* Opcode fetcher
|
|
****************************************************************************/
|
|
static CPU_READOP( tms32025 )
|
|
{
|
|
tms32025_state *cpustate = get_safe_token(device);
|
|
|
|
void *ptr;
|
|
|
|
/* skip if not custom */
|
|
if (!cpustate->pgmmap[offset >> 8])
|
|
return 0;
|
|
|
|
ptr = &((UINT8 *)&cpustate->pgmmap[offset >> 8])[offset & 0xff];
|
|
switch (size)
|
|
{
|
|
case 1: *value = *((UINT8 *) ptr);
|
|
case 2: *value = *((UINT16 *) ptr);
|
|
case 4: *value = *((UINT32 *) ptr);
|
|
case 8: *value = *((UINT64 *) ptr);
|
|
}
|
|
return 1;
|
|
}
|
|
|
|
|
|
/****************************************************************************
|
|
* Memory reader
|
|
****************************************************************************/
|
|
static CPU_READ( tms32025 )
|
|
{
|
|
tms32025_state *cpustate = get_safe_token(device);
|
|
|
|
void *ptr = NULL;
|
|
UINT64 temp = 0;
|
|
|
|
switch (space)
|
|
{
|
|
case AS_PROGRAM:
|
|
ptr = cpustate->pgmmap[offset >> 8];
|
|
if (!ptr)
|
|
return 0;
|
|
break;
|
|
|
|
case AS_DATA:
|
|
ptr = cpustate->datamap[offset >> 8];
|
|
if (!ptr)
|
|
return 0;
|
|
break;
|
|
|
|
default:
|
|
case AS_IO:
|
|
return 0;
|
|
}
|
|
|
|
switch (size)
|
|
{
|
|
case 1:
|
|
*value = ((UINT8 *)ptr)[BYTE_XOR_BE(offset & 0xff)];
|
|
break;
|
|
case 2:
|
|
*value = ((UINT16 *)ptr)[(offset & 0xff) / 2];
|
|
break;
|
|
case 4:
|
|
CPU_READ_NAME(tms32025)(device, space, offset + 0, 2, &temp);
|
|
*value = temp << 16;
|
|
CPU_READ_NAME(tms32025)(device, space, offset + 2, 2, &temp);
|
|
*value |= temp & 0xffff;
|
|
break;
|
|
case 8:
|
|
CPU_READ_NAME(tms32025)(device, space, offset + 0, 4, &temp);
|
|
*value = temp << 32;
|
|
CPU_READ_NAME(tms32025)(device, space, offset + 4, 4, &temp);
|
|
*value |= temp & 0xffffffff;
|
|
break;
|
|
}
|
|
return 1;
|
|
}
|
|
|
|
|
|
/****************************************************************************
|
|
* Memory writer
|
|
****************************************************************************/
|
|
static CPU_WRITE( tms32025 )
|
|
{
|
|
tms32025_state *cpustate = get_safe_token(device);
|
|
|
|
void *ptr = NULL;
|
|
|
|
switch (space)
|
|
{
|
|
case AS_PROGRAM:
|
|
ptr = cpustate->pgmmap[offset >> 8];
|
|
if (!ptr)
|
|
return 0;
|
|
break;
|
|
|
|
case AS_DATA:
|
|
ptr = cpustate->datamap[offset >> 8];
|
|
if (!ptr)
|
|
return 0;
|
|
break;
|
|
|
|
default:
|
|
case AS_IO:
|
|
return 0;
|
|
}
|
|
|
|
switch (size)
|
|
{
|
|
case 1:
|
|
((UINT8 *)ptr)[BYTE_XOR_BE(offset & 0xff)] = value;
|
|
break;
|
|
case 2:
|
|
((UINT16 *)ptr)[(offset & 0xff) / 2] = value;
|
|
break;
|
|
case 4:
|
|
CPU_WRITE_NAME(tms32025)(device, space, offset + 0, 2, value >> 16);
|
|
CPU_WRITE_NAME(tms32025)(device, space, offset + 2, 2, value);
|
|
break;
|
|
case 8:
|
|
CPU_WRITE_NAME(tms32025)(device, space, offset + 0, 4, value >> 32);
|
|
CPU_WRITE_NAME(tms32025)(device, space, offset + 4, 4, value);
|
|
break;
|
|
}
|
|
|
|
return 1;
|
|
}
|
|
|
|
|
|
/**************************************************************************
|
|
* Generic set_info
|
|
**************************************************************************/
|
|
|
|
static CPU_SET_INFO( tms32025 )
|
|
{
|
|
tms32025_state *cpustate = get_safe_token(device);
|
|
|
|
switch (state)
|
|
{
|
|
/* --- the following bits of info are set as 64-bit signed integers --- */
|
|
case CPUINFO_INT_INPUT_STATE + TMS32025_INT0: set_irq_line(cpustate, TMS32025_INT0, info->i); break;
|
|
case CPUINFO_INT_INPUT_STATE + TMS32025_INT1: set_irq_line(cpustate, TMS32025_INT1, info->i); break;
|
|
case CPUINFO_INT_INPUT_STATE + TMS32025_INT2: set_irq_line(cpustate, TMS32025_INT2, info->i); break;
|
|
case CPUINFO_INT_INPUT_STATE + TMS32025_TINT: set_irq_line(cpustate, TMS32025_TINT, info->i); break;
|
|
case CPUINFO_INT_INPUT_STATE + TMS32025_RINT: set_irq_line(cpustate, TMS32025_RINT, info->i); break;
|
|
case CPUINFO_INT_INPUT_STATE + TMS32025_XINT: set_irq_line(cpustate, TMS32025_XINT, info->i); break;
|
|
case CPUINFO_INT_INPUT_STATE + TMS32025_FSX: set_fsx_line(cpustate, info->i); break;
|
|
|
|
case CPUINFO_INT_PC:
|
|
case CPUINFO_INT_REGISTER + TMS32025_PC: cpustate->PC = info->i; break;
|
|
/* This is actually not a stack pointer, but the stack contents */
|
|
case CPUINFO_INT_SP:
|
|
case CPUINFO_INT_REGISTER + TMS32025_STK7: cpustate->STACK[7] = info->i; break;
|
|
case CPUINFO_INT_REGISTER + TMS32025_STK6: cpustate->STACK[6] = info->i; break;
|
|
case CPUINFO_INT_REGISTER + TMS32025_STK5: cpustate->STACK[5] = info->i; break;
|
|
case CPUINFO_INT_REGISTER + TMS32025_STK4: cpustate->STACK[4] = info->i; break;
|
|
case CPUINFO_INT_REGISTER + TMS32025_STK3: cpustate->STACK[3] = info->i; break;
|
|
case CPUINFO_INT_REGISTER + TMS32025_STK2: cpustate->STACK[2] = info->i; break;
|
|
case CPUINFO_INT_REGISTER + TMS32025_STK1: cpustate->STACK[1] = info->i; break;
|
|
case CPUINFO_INT_REGISTER + TMS32025_STK0: cpustate->STACK[0] = info->i; break;
|
|
case CPUINFO_INT_REGISTER + TMS32025_STR0: cpustate->STR0 = info->i; break;
|
|
case CPUINFO_INT_REGISTER + TMS32025_STR1: cpustate->STR1 = info->i; break;
|
|
case CPUINFO_INT_REGISTER + TMS32025_IFR: cpustate->IFR = info->i; break;
|
|
case CPUINFO_INT_REGISTER + TMS32025_RPTC: cpustate->RPTC = info->i; break;
|
|
case CPUINFO_INT_REGISTER + TMS32025_ACC: cpustate->ACC.d = info->i; break;
|
|
case CPUINFO_INT_REGISTER + TMS32025_PREG: cpustate->Preg.d = info->i; break;
|
|
case CPUINFO_INT_REGISTER + TMS32025_TREG: cpustate->Treg = info->i; break;
|
|
case CPUINFO_INT_REGISTER + TMS32025_AR0: cpustate->AR[0] = info->i; break;
|
|
case CPUINFO_INT_REGISTER + TMS32025_AR1: cpustate->AR[1] = info->i; break;
|
|
case CPUINFO_INT_REGISTER + TMS32025_AR2: cpustate->AR[2] = info->i; break;
|
|
case CPUINFO_INT_REGISTER + TMS32025_AR3: cpustate->AR[3] = info->i; break;
|
|
case CPUINFO_INT_REGISTER + TMS32025_AR4: cpustate->AR[4] = info->i; break;
|
|
case CPUINFO_INT_REGISTER + TMS32025_AR5: cpustate->AR[5] = info->i; break;
|
|
case CPUINFO_INT_REGISTER + TMS32025_AR6: cpustate->AR[6] = info->i; break;
|
|
case CPUINFO_INT_REGISTER + TMS32025_AR7: cpustate->AR[7] = info->i; break;
|
|
case CPUINFO_INT_REGISTER + TMS32025_DRR: M_WRTRAM(cpustate, 0,info->i); break;
|
|
case CPUINFO_INT_REGISTER + TMS32025_DXR: M_WRTRAM(cpustate, 1,info->i); break;
|
|
case CPUINFO_INT_REGISTER + TMS32025_TIM: M_WRTRAM(cpustate, 2,info->i); break;
|
|
case CPUINFO_INT_REGISTER + TMS32025_PRD: M_WRTRAM(cpustate, 3,info->i); break;
|
|
case CPUINFO_INT_REGISTER + TMS32025_IMR: M_WRTRAM(cpustate, 4,info->i); break;
|
|
case CPUINFO_INT_REGISTER + TMS32025_GREG: M_WRTRAM(cpustate, 5,info->i); break;
|
|
}
|
|
}
|
|
|
|
|
|
|
|
/**************************************************************************
|
|
* Generic get_info
|
|
**************************************************************************/
|
|
|
|
CPU_GET_INFO( tms32025 )
|
|
{
|
|
tms32025_state *cpustate = (device != NULL && device->token() != NULL) ? get_safe_token(device) : NULL;
|
|
|
|
switch (state)
|
|
{
|
|
/* --- the following bits of info are returned as 64-bit signed integers --- */
|
|
case CPUINFO_INT_CONTEXT_SIZE: info->i = sizeof(tms32025_state); break;
|
|
case CPUINFO_INT_INPUT_LINES: info->i = 6; break;
|
|
case CPUINFO_INT_DEFAULT_IRQ_VECTOR: info->i = 0; break;
|
|
case DEVINFO_INT_ENDIANNESS: info->i = ENDIANNESS_BIG; break;
|
|
case CPUINFO_INT_CLOCK_MULTIPLIER: info->i = 1; break;
|
|
case CPUINFO_INT_CLOCK_DIVIDER: info->i = 1; 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*CLK; break;
|
|
case CPUINFO_INT_MAX_CYCLES: info->i = 5*CLK; break;
|
|
|
|
case DEVINFO_INT_DATABUS_WIDTH + AS_PROGRAM: info->i = 16; break;
|
|
case DEVINFO_INT_ADDRBUS_WIDTH + AS_PROGRAM: info->i = 16; break;
|
|
case DEVINFO_INT_ADDRBUS_SHIFT + AS_PROGRAM: info->i = -1; break;
|
|
case DEVINFO_INT_DATABUS_WIDTH + AS_DATA: info->i = 16; break;
|
|
case DEVINFO_INT_ADDRBUS_WIDTH + AS_DATA: info->i = 16; break;
|
|
case DEVINFO_INT_ADDRBUS_SHIFT + AS_DATA: info->i = -1; break;
|
|
case DEVINFO_INT_DATABUS_WIDTH + AS_IO: info->i = 16; break;
|
|
case DEVINFO_INT_ADDRBUS_WIDTH + AS_IO: info->i = 17; break;
|
|
case DEVINFO_INT_ADDRBUS_SHIFT + AS_IO: info->i = -1; break;
|
|
|
|
case CPUINFO_INT_INPUT_STATE + TMS32025_INT0: info->i = (cpustate->IFR & 0x01) ? ASSERT_LINE : CLEAR_LINE; break;
|
|
case CPUINFO_INT_INPUT_STATE + TMS32025_INT1: info->i = (cpustate->IFR & 0x02) ? ASSERT_LINE : CLEAR_LINE; break;
|
|
case CPUINFO_INT_INPUT_STATE + TMS32025_INT2: info->i = (cpustate->IFR & 0x04) ? ASSERT_LINE : CLEAR_LINE; break;
|
|
case CPUINFO_INT_INPUT_STATE + TMS32025_TINT: info->i = (cpustate->IFR & 0x08) ? ASSERT_LINE : CLEAR_LINE; break;
|
|
case CPUINFO_INT_INPUT_STATE + TMS32025_RINT: info->i = (cpustate->IFR & 0x10) ? ASSERT_LINE : CLEAR_LINE; break;
|
|
case CPUINFO_INT_INPUT_STATE + TMS32025_XINT: info->i = (cpustate->IFR & 0x20) ? ASSERT_LINE : CLEAR_LINE; break;
|
|
|
|
case CPUINFO_INT_PREVIOUSPC: info->i = cpustate->PREVPC; break;
|
|
|
|
case CPUINFO_INT_PC:
|
|
case CPUINFO_INT_REGISTER + TMS32025_PC: info->i = cpustate->PC; break;
|
|
/* This is actually not a stack pointer, but the stack contents */
|
|
case CPUINFO_INT_SP:
|
|
case CPUINFO_INT_REGISTER + TMS32025_STK7: info->i = cpustate->STACK[7]; break;
|
|
case CPUINFO_INT_REGISTER + TMS32025_STK6: info->i = cpustate->STACK[6]; break;
|
|
case CPUINFO_INT_REGISTER + TMS32025_STK5: info->i = cpustate->STACK[5]; break;
|
|
case CPUINFO_INT_REGISTER + TMS32025_STK4: info->i = cpustate->STACK[4]; break;
|
|
case CPUINFO_INT_REGISTER + TMS32025_STK3: info->i = cpustate->STACK[3]; break;
|
|
case CPUINFO_INT_REGISTER + TMS32025_STK2: info->i = cpustate->STACK[2]; break;
|
|
case CPUINFO_INT_REGISTER + TMS32025_STK1: info->i = cpustate->STACK[1]; break;
|
|
case CPUINFO_INT_REGISTER + TMS32025_STK0: info->i = cpustate->STACK[0]; break;
|
|
case CPUINFO_INT_REGISTER + TMS32025_STR0: info->i = cpustate->STR0; break;
|
|
case CPUINFO_INT_REGISTER + TMS32025_STR1: info->i = cpustate->STR1; break;
|
|
case CPUINFO_INT_REGISTER + TMS32025_IFR: info->i = cpustate->IFR; break;
|
|
case CPUINFO_INT_REGISTER + TMS32025_RPTC: info->i = cpustate->RPTC; break;
|
|
case CPUINFO_INT_REGISTER + TMS32025_ACC: info->i = cpustate->ACC.d; break;
|
|
case CPUINFO_INT_REGISTER + TMS32025_PREG: info->i = cpustate->Preg.d; break;
|
|
case CPUINFO_INT_REGISTER + TMS32025_TREG: info->i = cpustate->Treg; break;
|
|
case CPUINFO_INT_REGISTER + TMS32025_AR0: info->i = cpustate->AR[0]; break;
|
|
case CPUINFO_INT_REGISTER + TMS32025_AR1: info->i = cpustate->AR[1]; break;
|
|
case CPUINFO_INT_REGISTER + TMS32025_AR2: info->i = cpustate->AR[2]; break;
|
|
case CPUINFO_INT_REGISTER + TMS32025_AR3: info->i = cpustate->AR[3]; break;
|
|
case CPUINFO_INT_REGISTER + TMS32025_AR4: info->i = cpustate->AR[4]; break;
|
|
case CPUINFO_INT_REGISTER + TMS32025_AR5: info->i = cpustate->AR[5]; break;
|
|
case CPUINFO_INT_REGISTER + TMS32025_AR6: info->i = cpustate->AR[6]; break;
|
|
case CPUINFO_INT_REGISTER + TMS32025_AR7: info->i = cpustate->AR[7]; break;
|
|
case CPUINFO_INT_REGISTER + TMS32025_DRR: info->i = M_RDRAM(cpustate, 0); break;
|
|
case CPUINFO_INT_REGISTER + TMS32025_DXR: info->i = M_RDRAM(cpustate, 1); break;
|
|
case CPUINFO_INT_REGISTER + TMS32025_TIM: info->i = M_RDRAM(cpustate, 2); break;
|
|
case CPUINFO_INT_REGISTER + TMS32025_PRD: info->i = M_RDRAM(cpustate, 3); break;
|
|
case CPUINFO_INT_REGISTER + TMS32025_IMR: info->i = M_RDRAM(cpustate, 4); break;
|
|
case CPUINFO_INT_REGISTER + TMS32025_GREG: info->i = M_RDRAM(cpustate, 5); break;
|
|
|
|
/* --- the following bits of info are returned as pointers to data or functions --- */
|
|
case CPUINFO_FCT_SET_INFO: info->setinfo = CPU_SET_INFO_NAME(tms32025); break;
|
|
case CPUINFO_FCT_INIT: info->init = CPU_INIT_NAME(tms32025); break;
|
|
case CPUINFO_FCT_RESET: info->reset = CPU_RESET_NAME(tms32025); break;
|
|
case CPUINFO_FCT_EXIT: info->exit = CPU_EXIT_NAME(tms32025); break;
|
|
case CPUINFO_FCT_EXECUTE: info->execute = CPU_EXECUTE_NAME(tms32025); break;
|
|
case CPUINFO_FCT_BURN: info->burn = NULL; break;
|
|
case CPUINFO_FCT_DISASSEMBLE: info->disassemble = CPU_DISASSEMBLE_NAME(tms32025); break;
|
|
case CPUINFO_FCT_READ: info->read = CPU_READ_NAME(tms32025); break;
|
|
case CPUINFO_FCT_WRITE: info->write = CPU_WRITE_NAME(tms32025); break;
|
|
case CPUINFO_FCT_READOP: info->readop = CPU_READOP_NAME(tms32025); break;
|
|
case CPUINFO_PTR_INSTRUCTION_COUNTER: info->icount = &cpustate->icount; break;
|
|
|
|
/* --- the following bits of info are returned as NULL-terminated strings --- */
|
|
case DEVINFO_STR_NAME: strcpy(info->s, "TMS32025"); break;
|
|
case DEVINFO_STR_FAMILY: strcpy(info->s, "Texas Instruments TMS320x25"); break;
|
|
case DEVINFO_STR_VERSION: strcpy(info->s, "1.10"); break;
|
|
case DEVINFO_STR_SOURCE_FILE: strcpy(info->s, __FILE__); break;
|
|
case DEVINFO_STR_CREDITS: strcpy(info->s, "Copyright Tony La Porta"); break;
|
|
|
|
case CPUINFO_STR_FLAGS:
|
|
sprintf(info->s, "arp%d%c%c%c%cdp%03x arb%d%c%c%c%c%c%c%c%c%c%c%cpm%d",
|
|
(cpustate->STR0 & 0xe000) >> 13,
|
|
cpustate->STR0 & 0x1000 ? 'O':'.',
|
|
cpustate->STR0 & 0x0800 ? 'M':'.',
|
|
cpustate->STR0 & 0x0400 ? '.':'?',
|
|
cpustate->STR0 & 0x0200 ? 'I':'.',
|
|
(cpustate->STR0 & 0x01ff),
|
|
|
|
(cpustate->STR1 & 0xe000) >> 13,
|
|
cpustate->STR1 & 0x1000 ? 'P':'D',
|
|
cpustate->STR1 & 0x0800 ? 'T':'.',
|
|
cpustate->STR1 & 0x0400 ? 'S':'.',
|
|
cpustate->STR1 & 0x0200 ? 'C':'?',
|
|
cpustate->STR0 & 0x0100 ? '.':'?',
|
|
cpustate->STR1 & 0x0080 ? '.':'?',
|
|
cpustate->STR1 & 0x0040 ? 'H':'.',
|
|
cpustate->STR1 & 0x0020 ? 'F':'.',
|
|
cpustate->STR1 & 0x0010 ? 'X':'.',
|
|
cpustate->STR1 & 0x0008 ? 'f':'.',
|
|
cpustate->STR1 & 0x0004 ? 'o':'i',
|
|
(cpustate->STR1 & 0x0003) );
|
|
break;
|
|
|
|
case CPUINFO_STR_REGISTER + TMS32025_PC: sprintf(info->s, "PC:%04X", cpustate->PC); break;
|
|
case CPUINFO_STR_REGISTER + TMS32025_STR0: sprintf(info->s, "STR0:%04X", cpustate->STR0); break;
|
|
case CPUINFO_STR_REGISTER + TMS32025_STR1: sprintf(info->s, "STR1:%04X", cpustate->STR1); break;
|
|
case CPUINFO_STR_REGISTER + TMS32025_IFR: sprintf(info->s, "IFR:%04X", cpustate->IFR); break;
|
|
case CPUINFO_STR_REGISTER + TMS32025_RPTC: sprintf(info->s, "RPTC:%02X", cpustate->RPTC); break;
|
|
case CPUINFO_STR_REGISTER + TMS32025_STK7: sprintf(info->s, "STK7:%04X", cpustate->STACK[7]); break;
|
|
case CPUINFO_STR_REGISTER + TMS32025_STK6: sprintf(info->s, "STK6:%04X", cpustate->STACK[6]); break;
|
|
case CPUINFO_STR_REGISTER + TMS32025_STK5: sprintf(info->s, "STK5:%04X", cpustate->STACK[5]); break;
|
|
case CPUINFO_STR_REGISTER + TMS32025_STK4: sprintf(info->s, "STK4:%04X", cpustate->STACK[4]); break;
|
|
case CPUINFO_STR_REGISTER + TMS32025_STK3: sprintf(info->s, "STK3:%04X", cpustate->STACK[3]); break;
|
|
case CPUINFO_STR_REGISTER + TMS32025_STK2: sprintf(info->s, "STK2:%04X", cpustate->STACK[2]); break;
|
|
case CPUINFO_STR_REGISTER + TMS32025_STK1: sprintf(info->s, "STK1:%04X", cpustate->STACK[1]); break;
|
|
case CPUINFO_STR_REGISTER + TMS32025_STK0: sprintf(info->s, "STK0:%04X", cpustate->STACK[0]); break;
|
|
case CPUINFO_STR_REGISTER + TMS32025_ACC: sprintf(info->s, "ACC:%08X", cpustate->ACC.d); break;
|
|
case CPUINFO_STR_REGISTER + TMS32025_PREG: sprintf(info->s, "P:%08X", cpustate->Preg.d); break;
|
|
case CPUINFO_STR_REGISTER + TMS32025_TREG: sprintf(info->s, "T:%04X", cpustate->Treg); break;
|
|
case CPUINFO_STR_REGISTER + TMS32025_AR0: sprintf(info->s, "AR0:%04X", cpustate->AR[0]); break;
|
|
case CPUINFO_STR_REGISTER + TMS32025_AR1: sprintf(info->s, "AR1:%04X", cpustate->AR[1]); break;
|
|
case CPUINFO_STR_REGISTER + TMS32025_AR2: sprintf(info->s, "AR2:%04X", cpustate->AR[2]); break;
|
|
case CPUINFO_STR_REGISTER + TMS32025_AR3: sprintf(info->s, "AR3:%04X", cpustate->AR[3]); break;
|
|
case CPUINFO_STR_REGISTER + TMS32025_AR4: sprintf(info->s, "AR4:%04X", cpustate->AR[4]); break;
|
|
case CPUINFO_STR_REGISTER + TMS32025_AR5: sprintf(info->s, "AR5:%04X", cpustate->AR[5]); break;
|
|
case CPUINFO_STR_REGISTER + TMS32025_AR6: sprintf(info->s, "AR6:%04X", cpustate->AR[6]); break;
|
|
case CPUINFO_STR_REGISTER + TMS32025_AR7: sprintf(info->s, "AR7:%04X", cpustate->AR[7]); break;
|
|
case CPUINFO_STR_REGISTER + TMS32025_DRR: sprintf(info->s, "DRR:%04X", M_RDRAM(cpustate, 0)); break;
|
|
case CPUINFO_STR_REGISTER + TMS32025_DXR: sprintf(info->s, "DXR:%04X", M_RDRAM(cpustate, 1)); break;
|
|
case CPUINFO_STR_REGISTER + TMS32025_TIM: sprintf(info->s, "TIM:%04X", M_RDRAM(cpustate, 2)); break;
|
|
case CPUINFO_STR_REGISTER + TMS32025_PRD: sprintf(info->s, "PRD:%04X", M_RDRAM(cpustate, 3)); break;
|
|
case CPUINFO_STR_REGISTER + TMS32025_IMR: sprintf(info->s, "IMR:%04X", M_RDRAM(cpustate, 4)); break;
|
|
case CPUINFO_STR_REGISTER + TMS32025_GREG: sprintf(info->s, "GREG:%04X", M_RDRAM(cpustate, 5)); break;
|
|
}
|
|
}
|
|
|
|
|
|
/**************************************************************************
|
|
* CPU-specific set_info
|
|
**************************************************************************/
|
|
|
|
CPU_GET_INFO( tms32026 )
|
|
{
|
|
switch (state)
|
|
{
|
|
/* --- the following bits of info are returned as pointers to data or functions --- */
|
|
case CPUINFO_FCT_RESET: info->reset = CPU_RESET_NAME(tms32026); break;
|
|
|
|
/* --- the following bits of info are returned as NULL-terminated strings --- */
|
|
case DEVINFO_STR_NAME: strcpy(info->s, "TMS32026"); break;
|
|
|
|
default: CPU_GET_INFO_CALL(tms32025); break;
|
|
}
|
|
}
|
|
|
|
DEFINE_LEGACY_CPU_DEVICE(TMS32025, tms32025);
|
|
DEFINE_LEGACY_CPU_DEVICE(TMS32026, tms32026);
|