mirror of
https://github.com/holub/mame
synced 2025-05-19 03:59:35 +03:00
2231 lines
54 KiB
C
2231 lines
54 KiB
C
/*===========================================================================
|
|
ASTAT -- ALU/MAC status register
|
|
===========================================================================*/
|
|
|
|
/* extracts flags */
|
|
#define GET_SS (m_astat & SSFLAG)
|
|
#define GET_MV (m_astat & MVFLAG)
|
|
#define GET_Q (m_astat & QFLAG)
|
|
#define GET_S (m_astat & SFLAG)
|
|
#define GET_C (m_astat & CFLAG)
|
|
#define GET_V (m_astat & VFLAG)
|
|
#define GET_N (m_astat & NFLAG)
|
|
#define GET_Z (m_astat & ZFLAG)
|
|
|
|
/* clears flags */
|
|
#define CLR_SS (m_astat &= ~SSFLAG)
|
|
#define CLR_MV (m_astat &= ~MVFLAG)
|
|
#define CLR_Q (m_astat &= ~QFLAG)
|
|
#define CLR_S (m_astat &= ~SFLAG)
|
|
#define CLR_C (m_astat &= ~CFLAG)
|
|
#define CLR_V (m_astat &= ~VFLAG)
|
|
#define CLR_N (m_astat &= ~NFLAG)
|
|
#define CLR_Z (m_astat &= ~ZFLAG)
|
|
|
|
/* sets flags */
|
|
#define SET_SS (m_astat |= SSFLAG)
|
|
#define SET_MV (m_astat |= MVFLAG)
|
|
#define SET_Q (m_astat |= QFLAG)
|
|
#define SET_S (m_astat |= SFLAG)
|
|
#define SET_C (m_astat |= CFLAG)
|
|
#define SET_V (m_astat |= VFLAG)
|
|
#define SET_Z (m_astat |= ZFLAG)
|
|
#define SET_N (m_astat |= NFLAG)
|
|
|
|
/* flag clearing; must be done before setting */
|
|
#define CLR_FLAGS (m_astat &= m_astat_clear)
|
|
|
|
/* compute flags */
|
|
#define CALC_Z(r) (m_astat |= ((r & 0xffff) == 0))
|
|
#define CALC_N(r) (m_astat |= (r >> 14) & 0x02)
|
|
#define CALC_V(s,d,r) (m_astat |= ((s ^ d ^ r ^ (r >> 1)) >> 13) & 0x04)
|
|
#define CALC_C(r) (m_astat |= (r >> 13) & 0x08)
|
|
#define CALC_C_SUB(r) (m_astat |= (~r >> 13) & 0x08)
|
|
#define CALC_NZ(r) CLR_FLAGS; CALC_N(r); CALC_Z(r)
|
|
#define CALC_NZV(s,d,r) CLR_FLAGS; CALC_N(r); CALC_Z(r); CALC_V(s,d,r)
|
|
#define CALC_NZVC(s,d,r) CLR_FLAGS; CALC_N(r); CALC_Z(r); CALC_V(s,d,r); CALC_C(r)
|
|
#define CALC_NZVC_SUB(s,d,r) CLR_FLAGS; CALC_N(r); CALC_Z(r); CALC_V(s,d,r); CALC_C_SUB(r)
|
|
|
|
/* ADSP-218x constants */
|
|
static const INT32 constants[] =
|
|
{
|
|
0x0001, 0xfffe, 0x0002, 0xfffd, 0x0004, 0xfffb, 0x0008, 0xfff7,
|
|
0x0010, 0xffef, 0x0020, 0xffdf, 0x0040, 0xffbf, 0x0080, 0xff7f,
|
|
0x0100, 0xfeff, 0x0200, 0xfdff, 0x0400, 0xfbff, 0x0800, 0xf7ff,
|
|
0x1000, 0xefff, 0x2000, 0xdfff, 0x4000, 0xbfff, 0x8000, 0x7fff
|
|
};
|
|
|
|
|
|
|
|
/*===========================================================================
|
|
MSTAT -- ALU/MAC control register
|
|
===========================================================================*/
|
|
|
|
/* flag definitions */
|
|
#define MSTAT_BANK 0x01 /* register bank select */
|
|
#define MSTAT_REVERSE 0x02 /* bit-reverse addressing enable (DAG1) */
|
|
#define MSTAT_STICKYV 0x04 /* sticky ALU overflow enable */
|
|
#define MSTAT_SATURATE 0x08 /* AR saturation mode enable */
|
|
#define MSTAT_INTEGER 0x10 /* MAC result placement; 0=fractional, 1=integer */
|
|
#define MSTAT_TIMER 0x20 /* timer enable */
|
|
#define MSTAT_GOMODE 0x40 /* go mode enable */
|
|
|
|
/* you must call this in order to change MSTAT */
|
|
inline void adsp21xx_device::update_mstat()
|
|
{
|
|
if ((m_mstat ^ m_mstat_prev) & MSTAT_BANK)
|
|
{
|
|
adsp_core temp = m_core;
|
|
m_core = m_alt;
|
|
m_alt = temp;
|
|
}
|
|
if ((m_mstat ^ m_mstat_prev) & MSTAT_TIMER)
|
|
if (m_timer_fired != NULL)
|
|
(*m_timer_fired)(*this, (m_mstat & MSTAT_TIMER) != 0);
|
|
if (m_mstat & MSTAT_STICKYV)
|
|
m_astat_clear = ~(CFLAG | NFLAG | ZFLAG);
|
|
else
|
|
m_astat_clear = ~(CFLAG | VFLAG | NFLAG | ZFLAG);
|
|
m_mstat_prev = m_mstat;
|
|
}
|
|
|
|
|
|
/*===========================================================================
|
|
SSTAT -- stack status register
|
|
===========================================================================*/
|
|
|
|
/* flag definitions */
|
|
#define PC_EMPTY 0x01 /* PC stack empty */
|
|
#define PC_OVER 0x02 /* PC stack overflow */
|
|
#define COUNT_EMPTY 0x04 /* count stack empty */
|
|
#define COUNT_OVER 0x08 /* count stack overflow */
|
|
#define STATUS_EMPTY 0x10 /* status stack empty */
|
|
#define STATUS_OVER 0x20 /* status stack overflow */
|
|
#define LOOP_EMPTY 0x40 /* loop stack empty */
|
|
#define LOOP_OVER 0x80 /* loop stack overflow */
|
|
|
|
|
|
|
|
/*===========================================================================
|
|
PC stack handlers
|
|
===========================================================================*/
|
|
|
|
inline UINT32 adsp21xx_device::pc_stack_top()
|
|
{
|
|
if (m_pc_sp > 0)
|
|
return m_pc_stack[m_pc_sp - 1];
|
|
else
|
|
return m_pc_stack[0];
|
|
}
|
|
|
|
inline void adsp21xx_device::set_pc_stack_top(UINT32 top)
|
|
{
|
|
if (m_pc_sp > 0)
|
|
m_pc_stack[m_pc_sp - 1] = top;
|
|
else
|
|
m_pc_stack[0] = top;
|
|
}
|
|
|
|
inline void adsp21xx_device::pc_stack_push()
|
|
{
|
|
if (m_pc_sp < PC_STACK_DEPTH)
|
|
{
|
|
m_pc_stack[m_pc_sp] = m_pc;
|
|
m_pc_sp++;
|
|
m_sstat &= ~PC_EMPTY;
|
|
}
|
|
else
|
|
m_sstat |= PC_OVER;
|
|
}
|
|
|
|
inline void adsp21xx_device::pc_stack_push_val(UINT32 val)
|
|
{
|
|
if (m_pc_sp < PC_STACK_DEPTH)
|
|
{
|
|
m_pc_stack[m_pc_sp] = val;
|
|
m_pc_sp++;
|
|
m_sstat &= ~PC_EMPTY;
|
|
}
|
|
else
|
|
m_sstat |= PC_OVER;
|
|
}
|
|
|
|
inline void adsp21xx_device::pc_stack_pop()
|
|
{
|
|
if (m_pc_sp > 0)
|
|
{
|
|
m_pc_sp--;
|
|
if (m_pc_sp == 0)
|
|
m_sstat |= PC_EMPTY;
|
|
}
|
|
m_pc = m_pc_stack[m_pc_sp];
|
|
}
|
|
|
|
inline UINT32 adsp21xx_device::pc_stack_pop_val()
|
|
{
|
|
if (m_pc_sp > 0)
|
|
{
|
|
m_pc_sp--;
|
|
if (m_pc_sp == 0)
|
|
m_sstat |= PC_EMPTY;
|
|
}
|
|
return m_pc_stack[m_pc_sp];
|
|
}
|
|
|
|
|
|
/*===========================================================================
|
|
CNTR stack handlers
|
|
===========================================================================*/
|
|
|
|
inline UINT32 adsp21xx_device::cntr_stack_top()
|
|
{
|
|
if (m_cntr_sp > 0)
|
|
return m_cntr_stack[m_cntr_sp - 1];
|
|
else
|
|
return m_cntr_stack[0];
|
|
}
|
|
|
|
inline void adsp21xx_device::cntr_stack_push()
|
|
{
|
|
if (m_cntr_sp < CNTR_STACK_DEPTH)
|
|
{
|
|
m_cntr_stack[m_cntr_sp] = m_cntr;
|
|
m_cntr_sp++;
|
|
m_sstat &= ~COUNT_EMPTY;
|
|
}
|
|
else
|
|
m_sstat |= COUNT_OVER;
|
|
}
|
|
|
|
inline void adsp21xx_device::cntr_stack_pop()
|
|
{
|
|
if (m_cntr_sp > 0)
|
|
{
|
|
m_cntr_sp--;
|
|
if (m_cntr_sp == 0)
|
|
m_sstat |= COUNT_EMPTY;
|
|
}
|
|
m_cntr = m_cntr_stack[m_cntr_sp];
|
|
}
|
|
|
|
|
|
/*===========================================================================
|
|
LOOP stack handlers
|
|
===========================================================================*/
|
|
|
|
inline UINT32 adsp21xx_device::loop_stack_top()
|
|
{
|
|
if (m_loop_sp > 0)
|
|
return m_loop_stack[m_loop_sp - 1];
|
|
else
|
|
return m_loop_stack[0];
|
|
}
|
|
|
|
inline void adsp21xx_device::loop_stack_push(UINT32 value)
|
|
{
|
|
if (m_loop_sp < LOOP_STACK_DEPTH)
|
|
{
|
|
m_loop_stack[m_loop_sp] = value;
|
|
m_loop_sp++;
|
|
m_loop = value >> 4;
|
|
m_loop_condition = value & 15;
|
|
m_sstat &= ~LOOP_EMPTY;
|
|
}
|
|
else
|
|
m_sstat |= LOOP_OVER;
|
|
}
|
|
|
|
inline void adsp21xx_device::loop_stack_pop()
|
|
{
|
|
if (m_loop_sp > 0)
|
|
{
|
|
m_loop_sp--;
|
|
if (m_loop_sp == 0)
|
|
{
|
|
m_loop = 0xffff;
|
|
m_loop_condition = 0;
|
|
m_sstat |= LOOP_EMPTY;
|
|
}
|
|
else
|
|
{
|
|
m_loop = m_loop_stack[m_loop_sp -1] >> 4;
|
|
m_loop_condition = m_loop_stack[m_loop_sp - 1] & 15;
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
/*===========================================================================
|
|
STAT stack handlers
|
|
===========================================================================*/
|
|
|
|
inline void adsp21xx_device::stat_stack_push()
|
|
{
|
|
if (m_stat_sp < STAT_STACK_DEPTH)
|
|
{
|
|
m_stat_stack[m_stat_sp][0] = m_mstat;
|
|
m_stat_stack[m_stat_sp][1] = m_imask;
|
|
m_stat_stack[m_stat_sp][2] = m_astat;
|
|
m_stat_sp++;
|
|
m_sstat &= ~STATUS_EMPTY;
|
|
}
|
|
else
|
|
m_sstat |= STATUS_OVER;
|
|
}
|
|
|
|
inline void adsp21xx_device::stat_stack_pop()
|
|
{
|
|
if (m_stat_sp > 0)
|
|
{
|
|
m_stat_sp--;
|
|
if (m_stat_sp == 0)
|
|
m_sstat |= STATUS_EMPTY;
|
|
}
|
|
m_mstat = m_stat_stack[m_stat_sp][0];
|
|
update_mstat();
|
|
m_imask = m_stat_stack[m_stat_sp][1];
|
|
m_astat = m_stat_stack[m_stat_sp][2];
|
|
check_irqs();
|
|
}
|
|
|
|
|
|
|
|
/*===========================================================================
|
|
condition code checking
|
|
===========================================================================*/
|
|
|
|
// gcc doesn't want to inline this, so we use a macro
|
|
#define condition(c) (((c) != 14) ? (m_condition_table[((c) << 8) | m_astat]) : slow_condition())
|
|
|
|
/*
|
|
inline int adsp21xx_device::condition(int c)
|
|
{
|
|
if (c != 14)
|
|
return m_condition_table[((c) << 8) | m_astat];
|
|
else
|
|
return slow_condition(c);
|
|
}
|
|
*/
|
|
|
|
int adsp21xx_device::slow_condition()
|
|
{
|
|
if ((INT32)--m_cntr > 0)
|
|
return 1;
|
|
else
|
|
{
|
|
cntr_stack_pop();
|
|
return 0;
|
|
}
|
|
}
|
|
|
|
|
|
|
|
/*===========================================================================
|
|
register writing
|
|
===========================================================================*/
|
|
|
|
inline void adsp21xx_device::update_i(int which)
|
|
{
|
|
m_base[which] = m_i[which] & m_lmask[which];
|
|
}
|
|
|
|
inline void adsp21xx_device::update_l(int which)
|
|
{
|
|
m_lmask[which] = m_mask_table[m_l[which] & 0x3fff];
|
|
m_base[which] = m_i[which] & m_lmask[which];
|
|
}
|
|
|
|
void adsp21xx_device::write_reg0(int regnum, INT32 val)
|
|
{
|
|
switch (regnum)
|
|
{
|
|
case 0x00: m_core.ax0.s = val; break;
|
|
case 0x01: m_core.ax1.s = val; break;
|
|
case 0x02: m_core.mx0.s = val; break;
|
|
case 0x03: m_core.mx1.s = val; break;
|
|
case 0x04: m_core.ay0.s = val; break;
|
|
case 0x05: m_core.ay1.s = val; break;
|
|
case 0x06: m_core.my0.s = val; break;
|
|
case 0x07: m_core.my1.s = val; break;
|
|
case 0x08: m_core.si.s = val; break;
|
|
case 0x09: m_core.se.s = (INT8)val; break;
|
|
case 0x0a: m_core.ar.s = val; break;
|
|
case 0x0b: m_core.mr.mrx.mr0.s = val; break;
|
|
case 0x0c: m_core.mr.mrx.mr1.s = val; m_core.mr.mrx.mr2.s = (INT16)val >> 15; break;
|
|
case 0x0d: m_core.mr.mrx.mr2.s = (INT8)val; break;
|
|
case 0x0e: m_core.sr.srx.sr0.s = val; break;
|
|
case 0x0f: m_core.sr.srx.sr1.s = val; break;
|
|
}
|
|
}
|
|
|
|
void adsp21xx_device::write_reg1(int regnum, INT32 val)
|
|
{
|
|
int index = regnum & 3;
|
|
switch (regnum >> 2)
|
|
{
|
|
case 0:
|
|
m_i[index] = val & 0x3fff;
|
|
update_i(index);
|
|
break;
|
|
|
|
case 1:
|
|
m_m[index] = (INT32)(val << 18) >> 18;
|
|
break;
|
|
|
|
case 2:
|
|
m_l[index] = val & 0x3fff;
|
|
update_l(index);
|
|
break;
|
|
|
|
case 3:
|
|
logerror("ADSP %04x: Writing to an invalid register!\n", m_ppc);
|
|
break;
|
|
}
|
|
}
|
|
|
|
void adsp21xx_device::write_reg2(int regnum, INT32 val)
|
|
{
|
|
int index = 4 + (regnum & 3);
|
|
switch (regnum >> 2)
|
|
{
|
|
case 0:
|
|
m_i[index] = val & 0x3fff;
|
|
update_i(index);
|
|
break;
|
|
|
|
case 1:
|
|
m_m[index] = (INT32)(val << 18) >> 18;
|
|
break;
|
|
|
|
case 2:
|
|
m_l[index] = val & 0x3fff;
|
|
update_l(index);
|
|
break;
|
|
|
|
case 3:
|
|
logerror("ADSP %04x: Writing to an invalid register!\n", m_ppc);
|
|
break;
|
|
}
|
|
}
|
|
|
|
void adsp21xx_device::write_reg3(int regnum, INT32 val)
|
|
{
|
|
switch (regnum)
|
|
{
|
|
case 0x00: m_astat = val & 0x00ff; break;
|
|
case 0x01: m_mstat = val & m_mstat_mask; update_mstat(); break;
|
|
case 0x03: m_imask = val & m_imask_mask; check_irqs(); break;
|
|
case 0x04: m_icntl = val & 0x001f; check_irqs(); break;
|
|
case 0x05: cntr_stack_push(); m_cntr = val & 0x3fff; break;
|
|
case 0x06: m_core.sb.s = (INT32)(val << 27) >> 27; break;
|
|
case 0x07: m_px = val; break;
|
|
case 0x09: if (m_sport_tx_callback != NULL) (*m_sport_tx_callback)(*this, 0, val); break;
|
|
case 0x0b: if (m_sport_tx_callback != NULL) (*m_sport_tx_callback)(*this, 1, val); break;
|
|
case 0x0c:
|
|
m_ifc = val;
|
|
if (m_chip_type >= CHIP_TYPE_ADSP2181)
|
|
{
|
|
/* clear timer */
|
|
if (val & 0x0002) m_irq_latch[ADSP2181_IRQ0] = 0;
|
|
if (val & 0x0004) m_irq_latch[ADSP2181_IRQ1] = 0;
|
|
/* clear BDMA */
|
|
if (val & 0x0010) m_irq_latch[ADSP2181_IRQE] = 0;
|
|
if (val & 0x0020) m_irq_latch[ADSP2181_SPORT0_RX] = 0;
|
|
if (val & 0x0040) m_irq_latch[ADSP2181_SPORT0_TX] = 0;
|
|
if (val & 0x0080) m_irq_latch[ADSP2181_IRQ2] = 0;
|
|
/* force timer */
|
|
if (val & 0x0200) m_irq_latch[ADSP2181_IRQ0] = 1;
|
|
if (val & 0x0400) m_irq_latch[ADSP2181_IRQ1] = 1;
|
|
/* force BDMA */
|
|
if (val & 0x1000) m_irq_latch[ADSP2181_IRQE] = 1;
|
|
if (val & 0x2000) m_irq_latch[ADSP2181_SPORT0_RX] = 1;
|
|
if (val & 0x4000) m_irq_latch[ADSP2181_SPORT0_TX] = 1;
|
|
if (val & 0x8000) m_irq_latch[ADSP2181_IRQ2] = 1;
|
|
}
|
|
else
|
|
{
|
|
/* clear timer */
|
|
if (val & 0x002) m_irq_latch[ADSP2101_IRQ0] = 0;
|
|
if (val & 0x004) m_irq_latch[ADSP2101_IRQ1] = 0;
|
|
if (val & 0x008) m_irq_latch[ADSP2101_SPORT0_RX] = 0;
|
|
if (val & 0x010) m_irq_latch[ADSP2101_SPORT0_TX] = 0;
|
|
if (val & 0x020) m_irq_latch[ADSP2101_IRQ2] = 0;
|
|
/* set timer */
|
|
if (val & 0x080) m_irq_latch[ADSP2101_IRQ0] = 1;
|
|
if (val & 0x100) m_irq_latch[ADSP2101_IRQ1] = 1;
|
|
if (val & 0x200) m_irq_latch[ADSP2101_SPORT0_RX] = 1;
|
|
if (val & 0x400) m_irq_latch[ADSP2101_SPORT0_TX] = 1;
|
|
if (val & 0x800) m_irq_latch[ADSP2101_IRQ2] = 1;
|
|
}
|
|
check_irqs();
|
|
break;
|
|
case 0x0d: m_cntr = val & 0x3fff; break;
|
|
case 0x0f: pc_stack_push_val(val & 0x3fff); break;
|
|
default: logerror("ADSP %04x: Writing to an invalid register!\n", m_ppc); break;
|
|
}
|
|
}
|
|
|
|
#define WRITE_REG(adsp,grp,reg,val) ((this->*wr_reg[grp][reg])(val))
|
|
|
|
|
|
|
|
/*===========================================================================
|
|
register reading
|
|
===========================================================================*/
|
|
|
|
INT32 adsp21xx_device::read_reg0(int regnum)
|
|
{
|
|
return *m_read0_ptr[regnum];
|
|
}
|
|
|
|
INT32 adsp21xx_device::read_reg1(int regnum)
|
|
{
|
|
return *m_read1_ptr[regnum];
|
|
}
|
|
|
|
INT32 adsp21xx_device::read_reg2(int regnum)
|
|
{
|
|
return *m_read2_ptr[regnum];
|
|
}
|
|
|
|
INT32 adsp21xx_device::read_reg3(int regnum)
|
|
{
|
|
switch (regnum)
|
|
{
|
|
case 0x00: return m_astat;
|
|
case 0x01: return m_mstat;
|
|
case 0x02: return m_sstat;
|
|
case 0x03: return m_imask;
|
|
case 0x04: return m_icntl;
|
|
case 0x05: return m_cntr;
|
|
case 0x06: return m_core.sb.s;
|
|
case 0x07: return m_px;
|
|
case 0x08: if (m_sport_rx_callback) return (*m_sport_rx_callback)(*this, 0); else return 0;
|
|
case 0x0a: if (m_sport_rx_callback) return (*m_sport_rx_callback)(*this, 1); else return 0;
|
|
case 0x0f: return pc_stack_pop_val();
|
|
default: logerror("ADSP %04x: Reading from an invalid register!\n", m_ppc); return 0;
|
|
}
|
|
}
|
|
|
|
|
|
|
|
/*===========================================================================
|
|
Modulus addressing logic
|
|
===========================================================================*/
|
|
|
|
inline void adsp21xx_device::modify_address(UINT32 ireg, UINT32 mreg)
|
|
{
|
|
UINT32 base = m_base[ireg];
|
|
UINT32 i = m_i[ireg];
|
|
UINT32 l = m_l[ireg];
|
|
|
|
i += m_m[mreg];
|
|
if (i < base) i += l;
|
|
else if (i >= base + l) i -= l;
|
|
m_i[ireg] = i;
|
|
}
|
|
|
|
|
|
|
|
/*===========================================================================
|
|
Data memory accessors
|
|
===========================================================================*/
|
|
|
|
inline void adsp21xx_device::data_write_dag1(UINT32 op, INT32 val)
|
|
{
|
|
UINT32 ireg = (op >> 2) & 3;
|
|
UINT32 mreg = op & 3;
|
|
UINT32 base = m_base[ireg];
|
|
UINT32 i = m_i[ireg];
|
|
UINT32 l = m_l[ireg];
|
|
|
|
if ( m_mstat & MSTAT_REVERSE )
|
|
{
|
|
UINT32 ir = m_reverse_table[ i & 0x3fff ];
|
|
data_write(ir, val);
|
|
}
|
|
else
|
|
data_write(i, val);
|
|
|
|
i += m_m[mreg];
|
|
if (i < base) i += l;
|
|
else if (i >= base + l) i -= l;
|
|
m_i[ireg] = i;
|
|
}
|
|
|
|
|
|
inline UINT32 adsp21xx_device::data_read_dag1(UINT32 op)
|
|
{
|
|
UINT32 ireg = (op >> 2) & 3;
|
|
UINT32 mreg = op & 3;
|
|
UINT32 base = m_base[ireg];
|
|
UINT32 i = m_i[ireg];
|
|
UINT32 l = m_l[ireg];
|
|
UINT32 res;
|
|
|
|
if (m_mstat & MSTAT_REVERSE)
|
|
{
|
|
UINT32 ir = m_reverse_table[i & 0x3fff];
|
|
res = data_read(ir);
|
|
}
|
|
else
|
|
res = data_read(i);
|
|
|
|
i += m_m[mreg];
|
|
if (i < base) i += l;
|
|
else if (i >= base + l) i -= l;
|
|
m_i[ireg] = i;
|
|
|
|
return res;
|
|
}
|
|
|
|
inline void adsp21xx_device::data_write_dag2(UINT32 op, INT32 val)
|
|
{
|
|
UINT32 ireg = 4 + ((op >> 2) & 3);
|
|
UINT32 mreg = 4 + (op & 3);
|
|
UINT32 base = m_base[ireg];
|
|
UINT32 i = m_i[ireg];
|
|
UINT32 l = m_l[ireg];
|
|
|
|
data_write(i, val);
|
|
|
|
i += m_m[mreg];
|
|
if (i < base) i += l;
|
|
else if (i >= base + l) i -= l;
|
|
m_i[ireg] = i;
|
|
}
|
|
|
|
|
|
inline UINT32 adsp21xx_device::data_read_dag2(UINT32 op)
|
|
{
|
|
UINT32 ireg = 4 + ((op >> 2) & 3);
|
|
UINT32 mreg = 4 + (op & 3);
|
|
UINT32 base = m_base[ireg];
|
|
UINT32 i = m_i[ireg];
|
|
UINT32 l = m_l[ireg];
|
|
|
|
UINT32 res = data_read(i);
|
|
|
|
i += m_m[mreg];
|
|
if (i < base) i += l;
|
|
else if (i >= base + l) i -= l;
|
|
m_i[ireg] = i;
|
|
|
|
return res;
|
|
}
|
|
|
|
/*===========================================================================
|
|
Program memory accessors
|
|
===========================================================================*/
|
|
|
|
inline void adsp21xx_device::pgm_write_dag2(UINT32 op, INT32 val)
|
|
{
|
|
UINT32 ireg = 4 + ((op >> 2) & 3);
|
|
UINT32 mreg = 4 + (op & 3);
|
|
UINT32 base = m_base[ireg];
|
|
UINT32 i = m_i[ireg];
|
|
UINT32 l = m_l[ireg];
|
|
|
|
program_write(i, (val << 8) | m_px);
|
|
|
|
i += m_m[mreg];
|
|
if (i < base) i += l;
|
|
else if (i >= base + l) i -= l;
|
|
m_i[ireg] = i;
|
|
}
|
|
|
|
|
|
inline UINT32 adsp21xx_device::pgm_read_dag2(UINT32 op)
|
|
{
|
|
UINT32 ireg = 4 + ((op >> 2) & 3);
|
|
UINT32 mreg = 4 + (op & 3);
|
|
UINT32 base = m_base[ireg];
|
|
UINT32 i = m_i[ireg];
|
|
UINT32 l = m_l[ireg];
|
|
UINT32 res;
|
|
|
|
res = program_read(i);
|
|
m_px = res;
|
|
res >>= 8;
|
|
|
|
i += m_m[mreg];
|
|
if (i < base) i += l;
|
|
else if (i >= base + l) i -= l;
|
|
m_i[ireg] = i;
|
|
|
|
return res;
|
|
}
|
|
|
|
|
|
|
|
/*===========================================================================
|
|
register reading
|
|
===========================================================================*/
|
|
|
|
#define ALU_GETXREG_UNSIGNED(x) (*(UINT16 *)m_alu_xregs[x])
|
|
#define ALU_GETYREG_UNSIGNED(y) (*(UINT16 *)m_alu_yregs[y])
|
|
|
|
#define MAC_GETXREG_UNSIGNED(x) (*(UINT16 *)m_mac_xregs[x])
|
|
#define MAC_GETXREG_SIGNED(x) (*( INT16 *)m_mac_xregs[x])
|
|
#define MAC_GETYREG_UNSIGNED(y) (*(UINT16 *)m_mac_yregs[y])
|
|
#define MAC_GETYREG_SIGNED(y) (*( INT16 *)m_mac_yregs[y])
|
|
|
|
#define SHIFT_GETXREG_UNSIGNED(x) (*(UINT16 *)m_shift_xregs[x])
|
|
#define SHIFT_GETXREG_SIGNED(x) (*( INT16 *)m_shift_xregs[x])
|
|
|
|
|
|
|
|
/*===========================================================================
|
|
ALU operations (result in AR)
|
|
===========================================================================*/
|
|
|
|
void adsp21xx_device::alu_op_ar(int op)
|
|
{
|
|
INT32 xop = (op >> 8) & 7;
|
|
INT32 yop = (op >> 11) & 3;
|
|
INT32 res;
|
|
|
|
switch (op & (15<<13)) /*JB*/
|
|
{
|
|
case 0x00<<13:
|
|
/* Y Clear when y = 0 */
|
|
res = ALU_GETYREG_UNSIGNED(yop);
|
|
CALC_NZ(res);
|
|
break;
|
|
case 0x01<<13:
|
|
/* Y + 1 PASS 1 when y = 0 */
|
|
yop = ALU_GETYREG_UNSIGNED(yop);
|
|
res = yop + 1;
|
|
CALC_NZ(res);
|
|
if (yop == 0x7fff) SET_V;
|
|
else if (yop == 0xffff) SET_C;
|
|
break;
|
|
case 0x02<<13:
|
|
/* X + Y + C */
|
|
xop = ALU_GETXREG_UNSIGNED(xop);
|
|
yop = ALU_GETYREG_UNSIGNED(yop);
|
|
yop += GET_C >> 3;
|
|
res = xop + yop;
|
|
CALC_NZVC(xop, yop, res);
|
|
break;
|
|
case 0x03<<13:
|
|
/* X + Y X when y = 0 */
|
|
xop = ALU_GETXREG_UNSIGNED(xop);
|
|
yop = ALU_GETYREG_UNSIGNED(yop);
|
|
res = xop + yop;
|
|
CALC_NZVC(xop, yop, res);
|
|
break;
|
|
case 0x04<<13:
|
|
/* NOT Y */
|
|
res = ALU_GETYREG_UNSIGNED(yop) ^ 0xffff;
|
|
CALC_NZ(res);
|
|
break;
|
|
case 0x05<<13:
|
|
/* -Y */
|
|
yop = ALU_GETYREG_UNSIGNED(yop);
|
|
res = -yop;
|
|
CALC_NZ(res);
|
|
if (yop == 0x8000) SET_V;
|
|
if (yop == 0x0000) SET_C;
|
|
break;
|
|
case 0x06<<13:
|
|
/* X - Y + C - 1 X + C - 1 when y = 0 */
|
|
xop = ALU_GETXREG_UNSIGNED(xop);
|
|
yop = ALU_GETYREG_UNSIGNED(yop);
|
|
res = xop - yop + (GET_C >> 3) - 1;
|
|
CALC_NZVC_SUB(xop, yop, res);
|
|
break;
|
|
case 0x07<<13:
|
|
/* X - Y */
|
|
xop = ALU_GETXREG_UNSIGNED(xop);
|
|
yop = ALU_GETYREG_UNSIGNED(yop);
|
|
res = xop - yop;
|
|
CALC_NZVC_SUB(xop, yop, res);
|
|
break;
|
|
case 0x08<<13:
|
|
/* Y - 1 PASS -1 when y = 0 */
|
|
yop = ALU_GETYREG_UNSIGNED(yop);
|
|
res = yop - 1;
|
|
CALC_NZ(res);
|
|
if (yop == 0x8000) SET_V;
|
|
else if (yop == 0x0000) SET_C;
|
|
break;
|
|
case 0x09<<13:
|
|
/* Y - X -X when y = 0 */
|
|
xop = ALU_GETXREG_UNSIGNED(xop);
|
|
yop = ALU_GETYREG_UNSIGNED(yop);
|
|
res = yop - xop;
|
|
CALC_NZVC_SUB(yop, xop, res);
|
|
break;
|
|
case 0x0a<<13:
|
|
/* Y - X + C - 1 -X + C - 1 when y = 0 */
|
|
xop = ALU_GETXREG_UNSIGNED(xop);
|
|
yop = ALU_GETYREG_UNSIGNED(yop);
|
|
res = yop - xop + (GET_C >> 3) - 1;
|
|
CALC_NZVC_SUB(yop, xop, res);
|
|
break;
|
|
case 0x0b<<13:
|
|
/* NOT X */
|
|
res = ALU_GETXREG_UNSIGNED(xop) ^ 0xffff;
|
|
CALC_NZ(res);
|
|
break;
|
|
case 0x0c<<13:
|
|
/* X AND Y */
|
|
xop = ALU_GETXREG_UNSIGNED(xop);
|
|
yop = ALU_GETYREG_UNSIGNED(yop);
|
|
res = xop & yop;
|
|
CALC_NZ(res);
|
|
break;
|
|
case 0x0d<<13:
|
|
/* X OR Y */
|
|
xop = ALU_GETXREG_UNSIGNED(xop);
|
|
yop = ALU_GETYREG_UNSIGNED(yop);
|
|
res = xop | yop;
|
|
CALC_NZ(res);
|
|
break;
|
|
case 0x0e<<13:
|
|
/* X XOR Y */
|
|
xop = ALU_GETXREG_UNSIGNED(xop);
|
|
yop = ALU_GETYREG_UNSIGNED(yop);
|
|
res = xop ^ yop;
|
|
CALC_NZ(res);
|
|
break;
|
|
case 0x0f<<13:
|
|
/* ABS X */
|
|
xop = ALU_GETXREG_UNSIGNED(xop);
|
|
res = (xop & 0x8000) ? -xop : xop;
|
|
CLR_FLAGS;
|
|
if (xop == 0) SET_Z;
|
|
if (xop == 0x8000) SET_N, SET_V;
|
|
if (xop & 0x8000) SET_S;
|
|
break;
|
|
default:
|
|
res = 0; /* just to keep the compiler happy */
|
|
break;
|
|
}
|
|
|
|
/* saturate */
|
|
if ((m_mstat & MSTAT_SATURATE) && GET_V) res = GET_C ? -32768 : 32767;
|
|
|
|
/* set the final value */
|
|
m_core.ar.u = res;
|
|
}
|
|
|
|
|
|
|
|
/*===========================================================================
|
|
ALU operations (result in AR, constant yop)
|
|
===========================================================================*/
|
|
|
|
void adsp21xx_device::alu_op_ar_const(int op)
|
|
{
|
|
INT32 xop = (op >> 8) & 7;
|
|
INT32 yop = constants[((op >> 5) & 0x07) | ((op >> 8) & 0x18)];
|
|
INT32 res;
|
|
|
|
switch (op & (15<<13)) /*JB*/
|
|
{
|
|
case 0x00<<13:
|
|
/* Y Clear when y = 0 */
|
|
res = yop;
|
|
CALC_NZ(res);
|
|
break;
|
|
case 0x01<<13:
|
|
/* Y + 1 PASS 1 when y = 0 */
|
|
res = yop + 1;
|
|
CALC_NZ(res);
|
|
if (yop == 0x7fff) SET_V;
|
|
else if (yop == 0xffff) SET_C;
|
|
break;
|
|
case 0x02<<13:
|
|
/* X + Y + C */
|
|
xop = ALU_GETXREG_UNSIGNED(xop);
|
|
yop += GET_C >> 3;
|
|
res = xop + yop;
|
|
CALC_NZVC(xop, yop, res);
|
|
break;
|
|
case 0x03<<13:
|
|
/* X + Y X when y = 0 */
|
|
xop = ALU_GETXREG_UNSIGNED(xop);
|
|
res = xop + yop;
|
|
CALC_NZVC(xop, yop, res);
|
|
break;
|
|
case 0x04<<13:
|
|
/* NOT Y */
|
|
res = yop ^ 0xffff;
|
|
CALC_NZ(res);
|
|
break;
|
|
case 0x05<<13:
|
|
/* -Y */
|
|
res = -yop;
|
|
CALC_NZ(res);
|
|
if (yop == 0x8000) SET_V;
|
|
if (yop == 0x0000) SET_C;
|
|
break;
|
|
case 0x06<<13:
|
|
/* X - Y + C - 1 X + C - 1 when y = 0 */
|
|
xop = ALU_GETXREG_UNSIGNED(xop);
|
|
res = xop - yop + (GET_C >> 3) - 1;
|
|
CALC_NZVC_SUB(xop, yop, res);
|
|
break;
|
|
case 0x07<<13:
|
|
/* X - Y */
|
|
xop = ALU_GETXREG_UNSIGNED(xop);
|
|
res = xop - yop;
|
|
CALC_NZVC_SUB(xop, yop, res);
|
|
break;
|
|
case 0x08<<13:
|
|
/* Y - 1 PASS -1 when y = 0 */
|
|
res = yop - 1;
|
|
CALC_NZ(res);
|
|
if (yop == 0x8000) SET_V;
|
|
else if (yop == 0x0000) SET_C;
|
|
break;
|
|
case 0x09<<13:
|
|
/* Y - X -X when y = 0 */
|
|
xop = ALU_GETXREG_UNSIGNED(xop);
|
|
res = yop - xop;
|
|
CALC_NZVC_SUB(yop, xop, res);
|
|
break;
|
|
case 0x0a<<13:
|
|
/* Y - X + C - 1 -X + C - 1 when y = 0 */
|
|
xop = ALU_GETXREG_UNSIGNED(xop);
|
|
res = yop - xop + (GET_C >> 3) - 1;
|
|
CALC_NZVC_SUB(yop, xop, res);
|
|
break;
|
|
case 0x0b<<13:
|
|
/* NOT X */
|
|
res = ALU_GETXREG_UNSIGNED(xop) ^ 0xffff;
|
|
CALC_NZ(res);
|
|
break;
|
|
case 0x0c<<13:
|
|
/* X AND Y */
|
|
xop = ALU_GETXREG_UNSIGNED(xop);
|
|
res = xop & yop;
|
|
CALC_NZ(res);
|
|
break;
|
|
case 0x0d<<13:
|
|
/* X OR Y */
|
|
xop = ALU_GETXREG_UNSIGNED(xop);
|
|
res = xop | yop;
|
|
CALC_NZ(res);
|
|
break;
|
|
case 0x0e<<13:
|
|
/* X XOR Y */
|
|
xop = ALU_GETXREG_UNSIGNED(xop);
|
|
res = xop ^ yop;
|
|
CALC_NZ(res);
|
|
break;
|
|
case 0x0f<<13:
|
|
/* ABS X */
|
|
xop = ALU_GETXREG_UNSIGNED(xop);
|
|
res = (xop & 0x8000) ? -xop : xop;
|
|
CLR_FLAGS;
|
|
if (xop == 0) SET_Z;
|
|
if (xop == 0x8000) SET_N, SET_V;
|
|
if (xop & 0x8000) SET_S;
|
|
break;
|
|
default:
|
|
res = 0; /* just to keep the compiler happy */
|
|
break;
|
|
}
|
|
|
|
/* saturate */
|
|
if ((m_mstat & MSTAT_SATURATE) && GET_V) res = GET_C ? -32768 : 32767;
|
|
|
|
/* set the final value */
|
|
m_core.ar.u = res;
|
|
}
|
|
|
|
|
|
|
|
/*===========================================================================
|
|
ALU operations (result in AF)
|
|
===========================================================================*/
|
|
|
|
void adsp21xx_device::alu_op_af(int op)
|
|
{
|
|
INT32 xop = (op >> 8) & 7;
|
|
INT32 yop = (op >> 11) & 3;
|
|
INT32 res;
|
|
|
|
switch (op & (15<<13)) /*JB*/
|
|
{
|
|
case 0x00<<13:
|
|
/* Y Clear when y = 0 */
|
|
res = ALU_GETYREG_UNSIGNED(yop);
|
|
CALC_NZ(res);
|
|
break;
|
|
case 0x01<<13:
|
|
/* Y + 1 PASS 1 when y = 0 */
|
|
yop = ALU_GETYREG_UNSIGNED(yop);
|
|
res = yop + 1;
|
|
CALC_NZ(res);
|
|
if (yop == 0x7fff) SET_V;
|
|
else if (yop == 0xffff) SET_C;
|
|
break;
|
|
case 0x02<<13:
|
|
/* X + Y + C */
|
|
xop = ALU_GETXREG_UNSIGNED(xop);
|
|
yop = ALU_GETYREG_UNSIGNED(yop);
|
|
yop += GET_C >> 3;
|
|
res = xop + yop;
|
|
CALC_NZVC(xop, yop, res);
|
|
break;
|
|
case 0x03<<13:
|
|
/* X + Y X when y = 0 */
|
|
xop = ALU_GETXREG_UNSIGNED(xop);
|
|
yop = ALU_GETYREG_UNSIGNED(yop);
|
|
res = xop + yop;
|
|
CALC_NZVC(xop, yop, res);
|
|
break;
|
|
case 0x04<<13:
|
|
/* NOT Y */
|
|
res = ALU_GETYREG_UNSIGNED(yop) ^ 0xffff;
|
|
CALC_NZ(res);
|
|
break;
|
|
case 0x05<<13:
|
|
/* -Y */
|
|
yop = ALU_GETYREG_UNSIGNED(yop);
|
|
res = -yop;
|
|
CALC_NZ(res);
|
|
if (yop == 0x8000) SET_V;
|
|
if (yop == 0x0000) SET_C;
|
|
break;
|
|
case 0x06<<13:
|
|
/* X - Y + C - 1 X + C - 1 when y = 0 */
|
|
xop = ALU_GETXREG_UNSIGNED(xop);
|
|
yop = ALU_GETYREG_UNSIGNED(yop);
|
|
res = xop - yop + (GET_C >> 3) - 1;
|
|
CALC_NZVC_SUB(xop, yop, res);
|
|
break;
|
|
case 0x07<<13:
|
|
/* X - Y */
|
|
xop = ALU_GETXREG_UNSIGNED(xop);
|
|
yop = ALU_GETYREG_UNSIGNED(yop);
|
|
res = xop - yop;
|
|
CALC_NZVC_SUB(xop, yop, res);
|
|
break;
|
|
case 0x08<<13:
|
|
/* Y - 1 PASS -1 when y = 0 */
|
|
yop = ALU_GETYREG_UNSIGNED(yop);
|
|
res = yop - 1;
|
|
CALC_NZ(res);
|
|
if (yop == 0x8000) SET_V;
|
|
else if (yop == 0x0000) SET_C;
|
|
break;
|
|
case 0x09<<13:
|
|
/* Y - X -X when y = 0 */
|
|
xop = ALU_GETXREG_UNSIGNED(xop);
|
|
yop = ALU_GETYREG_UNSIGNED(yop);
|
|
res = yop - xop;
|
|
CALC_NZVC_SUB(yop, xop, res);
|
|
break;
|
|
case 0x0a<<13:
|
|
/* Y - X + C - 1 -X + C - 1 when y = 0 */
|
|
xop = ALU_GETXREG_UNSIGNED(xop);
|
|
yop = ALU_GETYREG_UNSIGNED(yop);
|
|
res = yop - xop + (GET_C >> 3) - 1;
|
|
CALC_NZVC_SUB(yop, xop, res);
|
|
break;
|
|
case 0x0b<<13:
|
|
/* NOT X */
|
|
res = ALU_GETXREG_UNSIGNED(xop) ^ 0xffff;
|
|
CALC_NZ(res);
|
|
break;
|
|
case 0x0c<<13:
|
|
/* X AND Y */
|
|
xop = ALU_GETXREG_UNSIGNED(xop);
|
|
yop = ALU_GETYREG_UNSIGNED(yop);
|
|
res = xop & yop;
|
|
CALC_NZ(res);
|
|
break;
|
|
case 0x0d<<13:
|
|
/* X OR Y */
|
|
xop = ALU_GETXREG_UNSIGNED(xop);
|
|
yop = ALU_GETYREG_UNSIGNED(yop);
|
|
res = xop | yop;
|
|
CALC_NZ(res);
|
|
break;
|
|
case 0x0e<<13:
|
|
/* X XOR Y */
|
|
xop = ALU_GETXREG_UNSIGNED(xop);
|
|
yop = ALU_GETYREG_UNSIGNED(yop);
|
|
res = xop ^ yop;
|
|
CALC_NZ(res);
|
|
break;
|
|
case 0x0f<<13:
|
|
/* ABS X */
|
|
xop = ALU_GETXREG_UNSIGNED(xop);
|
|
res = (xop & 0x8000) ? -xop : xop;
|
|
CLR_FLAGS;
|
|
if (xop == 0) SET_Z;
|
|
if (xop == 0x8000) SET_N, SET_V;
|
|
if (xop & 0x8000) SET_S;
|
|
break;
|
|
default:
|
|
res = 0; /* just to keep the compiler happy */
|
|
break;
|
|
}
|
|
|
|
/* set the final value */
|
|
m_core.af.u = res;
|
|
}
|
|
|
|
|
|
|
|
/*===========================================================================
|
|
ALU operations (result in AF, constant yop)
|
|
===========================================================================*/
|
|
|
|
void adsp21xx_device::alu_op_af_const(int op)
|
|
{
|
|
INT32 xop = (op >> 8) & 7;
|
|
INT32 yop = constants[((op >> 5) & 0x07) | ((op >> 8) & 0x18)];
|
|
INT32 res;
|
|
|
|
switch (op & (15<<13)) /*JB*/
|
|
{
|
|
case 0x00<<13:
|
|
/* Y Clear when y = 0 */
|
|
res = yop;
|
|
CALC_NZ(res);
|
|
break;
|
|
case 0x01<<13:
|
|
/* Y + 1 PASS 1 when y = 0 */
|
|
res = yop + 1;
|
|
CALC_NZ(res);
|
|
if (yop == 0x7fff) SET_V;
|
|
else if (yop == 0xffff) SET_C;
|
|
break;
|
|
case 0x02<<13:
|
|
/* X + Y + C */
|
|
xop = ALU_GETXREG_UNSIGNED(xop);
|
|
yop += GET_C >> 3;
|
|
res = xop + yop;
|
|
CALC_NZVC(xop, yop, res);
|
|
break;
|
|
case 0x03<<13:
|
|
/* X + Y X when y = 0 */
|
|
xop = ALU_GETXREG_UNSIGNED(xop);
|
|
res = xop + yop;
|
|
CALC_NZVC(xop, yop, res);
|
|
break;
|
|
case 0x04<<13:
|
|
/* NOT Y */
|
|
res = yop ^ 0xffff;
|
|
CALC_NZ(res);
|
|
break;
|
|
case 0x05<<13:
|
|
/* -Y */
|
|
res = -yop;
|
|
CALC_NZ(res);
|
|
if (yop == 0x8000) SET_V;
|
|
if (yop == 0x0000) SET_C;
|
|
break;
|
|
case 0x06<<13:
|
|
/* X - Y + C - 1 X + C - 1 when y = 0 */
|
|
xop = ALU_GETXREG_UNSIGNED(xop);
|
|
res = xop - yop + (GET_C >> 3) - 1;
|
|
CALC_NZVC_SUB(xop, yop, res);
|
|
break;
|
|
case 0x07<<13:
|
|
/* X - Y */
|
|
xop = ALU_GETXREG_UNSIGNED(xop);
|
|
res = xop - yop;
|
|
CALC_NZVC_SUB(xop, yop, res);
|
|
break;
|
|
case 0x08<<13:
|
|
/* Y - 1 PASS -1 when y = 0 */
|
|
res = yop - 1;
|
|
CALC_NZ(res);
|
|
if (yop == 0x8000) SET_V;
|
|
else if (yop == 0x0000) SET_C;
|
|
break;
|
|
case 0x09<<13:
|
|
/* Y - X -X when y = 0 */
|
|
xop = ALU_GETXREG_UNSIGNED(xop);
|
|
res = yop - xop;
|
|
CALC_NZVC_SUB(yop, xop, res);
|
|
break;
|
|
case 0x0a<<13:
|
|
/* Y - X + C - 1 -X + C - 1 when y = 0 */
|
|
xop = ALU_GETXREG_UNSIGNED(xop);
|
|
res = yop - xop + (GET_C >> 3) - 1;
|
|
CALC_NZVC_SUB(yop, xop, res);
|
|
break;
|
|
case 0x0b<<13:
|
|
/* NOT X */
|
|
res = ALU_GETXREG_UNSIGNED(xop) ^ 0xffff;
|
|
CALC_NZ(res);
|
|
break;
|
|
case 0x0c<<13:
|
|
/* X AND Y */
|
|
xop = ALU_GETXREG_UNSIGNED(xop);
|
|
res = xop & yop;
|
|
CALC_NZ(res);
|
|
break;
|
|
case 0x0d<<13:
|
|
/* X OR Y */
|
|
xop = ALU_GETXREG_UNSIGNED(xop);
|
|
res = xop | yop;
|
|
CALC_NZ(res);
|
|
break;
|
|
case 0x0e<<13:
|
|
/* X XOR Y */
|
|
xop = ALU_GETXREG_UNSIGNED(xop);
|
|
res = xop ^ yop;
|
|
CALC_NZ(res);
|
|
break;
|
|
case 0x0f<<13:
|
|
/* ABS X */
|
|
xop = ALU_GETXREG_UNSIGNED(xop);
|
|
res = (xop & 0x8000) ? -xop : xop;
|
|
CLR_FLAGS;
|
|
if (xop == 0) SET_Z;
|
|
if (xop == 0x8000) SET_N, SET_V;
|
|
if (xop & 0x8000) SET_S;
|
|
break;
|
|
default:
|
|
res = 0; /* just to keep the compiler happy */
|
|
break;
|
|
}
|
|
|
|
/* set the final value */
|
|
m_core.af.u = res;
|
|
}
|
|
|
|
|
|
|
|
/*===========================================================================
|
|
ALU operations (no result)
|
|
===========================================================================*/
|
|
|
|
void adsp21xx_device::alu_op_none(int op)
|
|
{
|
|
INT32 xop = (op >> 8) & 7;
|
|
INT32 yop = (op >> 11) & 3;
|
|
INT32 res;
|
|
|
|
switch (op & (15<<13)) /*JB*/
|
|
{
|
|
case 0x00<<13:
|
|
/* Y Clear when y = 0 */
|
|
res = ALU_GETYREG_UNSIGNED(yop);
|
|
CALC_NZ(res);
|
|
break;
|
|
case 0x01<<13:
|
|
/* Y + 1 PASS 1 when y = 0 */
|
|
yop = ALU_GETYREG_UNSIGNED(yop);
|
|
res = yop + 1;
|
|
CALC_NZ(res);
|
|
if (yop == 0x7fff) SET_V;
|
|
else if (yop == 0xffff) SET_C;
|
|
break;
|
|
case 0x02<<13:
|
|
/* X + Y + C */
|
|
xop = ALU_GETXREG_UNSIGNED(xop);
|
|
yop = ALU_GETYREG_UNSIGNED(yop);
|
|
yop += GET_C >> 3;
|
|
res = xop + yop;
|
|
CALC_NZVC(xop, yop, res);
|
|
break;
|
|
case 0x03<<13:
|
|
/* X + Y X when y = 0 */
|
|
xop = ALU_GETXREG_UNSIGNED(xop);
|
|
yop = ALU_GETYREG_UNSIGNED(yop);
|
|
res = xop + yop;
|
|
CALC_NZVC(xop, yop, res);
|
|
break;
|
|
case 0x04<<13:
|
|
/* NOT Y */
|
|
res = ALU_GETYREG_UNSIGNED(yop) ^ 0xffff;
|
|
CALC_NZ(res);
|
|
break;
|
|
case 0x05<<13:
|
|
/* -Y */
|
|
yop = ALU_GETYREG_UNSIGNED(yop);
|
|
res = -yop;
|
|
CALC_NZ(res);
|
|
if (yop == 0x8000) SET_V;
|
|
if (yop == 0x0000) SET_C;
|
|
break;
|
|
case 0x06<<13:
|
|
/* X - Y + C - 1 X + C - 1 when y = 0 */
|
|
xop = ALU_GETXREG_UNSIGNED(xop);
|
|
yop = ALU_GETYREG_UNSIGNED(yop);
|
|
res = xop - yop + (GET_C >> 3) - 1;
|
|
CALC_NZVC_SUB(xop, yop, res);
|
|
break;
|
|
case 0x07<<13:
|
|
/* X - Y */
|
|
xop = ALU_GETXREG_UNSIGNED(xop);
|
|
yop = ALU_GETYREG_UNSIGNED(yop);
|
|
res = xop - yop;
|
|
CALC_NZVC_SUB(xop, yop, res);
|
|
break;
|
|
case 0x08<<13:
|
|
/* Y - 1 PASS -1 when y = 0 */
|
|
yop = ALU_GETYREG_UNSIGNED(yop);
|
|
res = yop - 1;
|
|
CALC_NZ(res);
|
|
if (yop == 0x8000) SET_V;
|
|
else if (yop == 0x0000) SET_C;
|
|
break;
|
|
case 0x09<<13:
|
|
/* Y - X -X when y = 0 */
|
|
xop = ALU_GETXREG_UNSIGNED(xop);
|
|
yop = ALU_GETYREG_UNSIGNED(yop);
|
|
res = yop - xop;
|
|
CALC_NZVC_SUB(yop, xop, res);
|
|
break;
|
|
case 0x0a<<13:
|
|
/* Y - X + C - 1 -X + C - 1 when y = 0 */
|
|
xop = ALU_GETXREG_UNSIGNED(xop);
|
|
yop = ALU_GETYREG_UNSIGNED(yop);
|
|
res = yop - xop + (GET_C >> 3) - 1;
|
|
CALC_NZVC_SUB(yop, xop, res);
|
|
break;
|
|
case 0x0b<<13:
|
|
/* NOT X */
|
|
res = ALU_GETXREG_UNSIGNED(xop) ^ 0xffff;
|
|
CALC_NZ(res);
|
|
break;
|
|
case 0x0c<<13:
|
|
/* X AND Y */
|
|
xop = ALU_GETXREG_UNSIGNED(xop);
|
|
yop = ALU_GETYREG_UNSIGNED(yop);
|
|
res = xop & yop;
|
|
CALC_NZ(res);
|
|
break;
|
|
case 0x0d<<13:
|
|
/* X OR Y */
|
|
xop = ALU_GETXREG_UNSIGNED(xop);
|
|
yop = ALU_GETYREG_UNSIGNED(yop);
|
|
res = xop | yop;
|
|
CALC_NZ(res);
|
|
break;
|
|
case 0x0e<<13:
|
|
/* X XOR Y */
|
|
xop = ALU_GETXREG_UNSIGNED(xop);
|
|
yop = ALU_GETYREG_UNSIGNED(yop);
|
|
res = xop ^ yop;
|
|
CALC_NZ(res);
|
|
break;
|
|
case 0x0f<<13:
|
|
/* ABS X */
|
|
xop = ALU_GETXREG_UNSIGNED(xop);
|
|
res = (xop & 0x8000) ? -xop : xop;
|
|
CLR_FLAGS;
|
|
if (xop == 0) SET_Z;
|
|
if (xop == 0x8000) SET_N, SET_V;
|
|
if (xop & 0x8000) SET_S;
|
|
break;
|
|
}
|
|
}
|
|
|
|
|
|
|
|
/*===========================================================================
|
|
MAC operations (result in MR)
|
|
===========================================================================*/
|
|
|
|
void adsp21xx_device::mac_op_mr(int op)
|
|
{
|
|
INT8 shift = ((m_mstat & MSTAT_INTEGER) >> 4) ^ 1;
|
|
INT32 xop = (op >> 8) & 7;
|
|
INT32 yop = (op >> 11) & 3;
|
|
INT32 temp;
|
|
INT64 res;
|
|
|
|
switch (op & (15<<13)) /*JB*/
|
|
{
|
|
case 0x00<<13:
|
|
/* no-op */
|
|
return;
|
|
case 0x01<<13:
|
|
/* X * Y (RND) */
|
|
xop = MAC_GETXREG_SIGNED(xop);
|
|
yop = MAC_GETYREG_SIGNED(yop);
|
|
temp = (xop * yop) << shift;
|
|
res = (INT64)temp;
|
|
#if 0
|
|
if ((res & 0xffff) == 0x8000) res &= ~((UINT64)0x10000);
|
|
else res += (res & 0x8000) << 1;
|
|
#else
|
|
temp &= 0xffff;
|
|
res += 0x8000;
|
|
if ( temp == 0x8000 )
|
|
res &= ~((UINT64)0x10000);
|
|
#endif
|
|
break;
|
|
case 0x02<<13:
|
|
/* MR + X * Y (RND) */
|
|
xop = MAC_GETXREG_SIGNED(xop);
|
|
yop = MAC_GETYREG_SIGNED(yop);
|
|
temp = (xop * yop) << shift;
|
|
res = m_core.mr.mr + (INT64)temp;
|
|
#if 0
|
|
if ((res & 0xffff) == 0x8000) res &= ~((UINT64)0x10000);
|
|
else res += (res & 0x8000) << 1;
|
|
#else
|
|
temp &= 0xffff;
|
|
res += 0x8000;
|
|
if ( temp == 0x8000 )
|
|
res &= ~((UINT64)0x10000);
|
|
#endif
|
|
break;
|
|
case 0x03<<13:
|
|
/* MR - X * Y (RND) */
|
|
xop = MAC_GETXREG_SIGNED(xop);
|
|
yop = MAC_GETYREG_SIGNED(yop);
|
|
temp = (xop * yop) << shift;
|
|
res = m_core.mr.mr - (INT64)temp;
|
|
#if 0
|
|
if ((res & 0xffff) == 0x8000) res &= ~((UINT64)0x10000);
|
|
else res += (res & 0x8000) << 1;
|
|
#else
|
|
temp &= 0xffff;
|
|
res += 0x8000;
|
|
if ( temp == 0x8000 )
|
|
res &= ~((UINT64)0x10000);
|
|
#endif
|
|
break;
|
|
case 0x04<<13:
|
|
/* X * Y (SS) Clear when y = 0 */
|
|
xop = MAC_GETXREG_SIGNED(xop);
|
|
yop = MAC_GETYREG_SIGNED(yop);
|
|
temp = (xop * yop) << shift;
|
|
res = (INT64)temp;
|
|
break;
|
|
case 0x05<<13:
|
|
/* X * Y (SU) */
|
|
xop = MAC_GETXREG_SIGNED(xop);
|
|
yop = MAC_GETYREG_UNSIGNED(yop);
|
|
temp = (xop * yop) << shift;
|
|
res = (INT64)temp;
|
|
break;
|
|
case 0x06<<13:
|
|
/* X * Y (US) */
|
|
xop = MAC_GETXREG_UNSIGNED(xop);
|
|
yop = MAC_GETYREG_SIGNED(yop);
|
|
temp = (xop * yop) << shift;
|
|
res = (INT64)temp;
|
|
break;
|
|
case 0x07<<13:
|
|
/* X * Y (UU) */
|
|
xop = MAC_GETXREG_UNSIGNED(xop);
|
|
yop = MAC_GETYREG_UNSIGNED(yop);
|
|
temp = (xop * yop) << shift;
|
|
res = (INT64)temp;
|
|
break;
|
|
case 0x08<<13:
|
|
/* MR + X * Y (SS) */
|
|
xop = MAC_GETXREG_SIGNED(xop);
|
|
yop = MAC_GETYREG_SIGNED(yop);
|
|
temp = (xop * yop) << shift;
|
|
res = m_core.mr.mr + (INT64)temp;
|
|
break;
|
|
case 0x09<<13:
|
|
/* MR + X * Y (SU) */
|
|
xop = MAC_GETXREG_SIGNED(xop);
|
|
yop = MAC_GETYREG_UNSIGNED(yop);
|
|
temp = (xop * yop) << shift;
|
|
res = m_core.mr.mr + (INT64)temp;
|
|
break;
|
|
case 0x0a<<13:
|
|
/* MR + X * Y (US) */
|
|
xop = MAC_GETXREG_UNSIGNED(xop);
|
|
yop = MAC_GETYREG_SIGNED(yop);
|
|
temp = (xop * yop) << shift;
|
|
res = m_core.mr.mr + (INT64)temp;
|
|
break;
|
|
case 0x0b<<13:
|
|
/* MR + X * Y (UU) */
|
|
xop = MAC_GETXREG_UNSIGNED(xop);
|
|
yop = MAC_GETYREG_UNSIGNED(yop);
|
|
temp = (xop * yop) << shift;
|
|
res = m_core.mr.mr + (INT64)temp;
|
|
break;
|
|
case 0x0c<<13:
|
|
/* MR - X * Y (SS) */
|
|
xop = MAC_GETXREG_SIGNED(xop);
|
|
yop = MAC_GETYREG_SIGNED(yop);
|
|
temp = (xop * yop) << shift;
|
|
res = m_core.mr.mr - (INT64)temp;
|
|
break;
|
|
case 0x0d<<13:
|
|
/* MR - X * Y (SU) */
|
|
xop = MAC_GETXREG_SIGNED(xop);
|
|
yop = MAC_GETYREG_UNSIGNED(yop);
|
|
temp = (xop * yop) << shift;
|
|
res = m_core.mr.mr - (INT64)temp;
|
|
break;
|
|
case 0x0e<<13:
|
|
/* MR - X * Y (US) */
|
|
xop = MAC_GETXREG_UNSIGNED(xop);
|
|
yop = MAC_GETYREG_SIGNED(yop);
|
|
temp = (xop * yop) << shift;
|
|
res = m_core.mr.mr - (INT64)temp;
|
|
break;
|
|
case 0x0f<<13:
|
|
/* MR - X * Y (UU) */
|
|
xop = MAC_GETXREG_UNSIGNED(xop);
|
|
yop = MAC_GETYREG_UNSIGNED(yop);
|
|
temp = (xop * yop) << shift;
|
|
res = m_core.mr.mr - (INT64)temp;
|
|
break;
|
|
default:
|
|
res = 0; /* just to keep the compiler happy */
|
|
break;
|
|
}
|
|
|
|
/* set the final value */
|
|
temp = (res >> 31) & 0x1ff;
|
|
CLR_MV;
|
|
if (temp != 0x000 && temp != 0x1ff) SET_MV;
|
|
m_core.mr.mr = res;
|
|
}
|
|
|
|
|
|
|
|
/*===========================================================================
|
|
MAC operations (result in MR, yop == xop)
|
|
===========================================================================*/
|
|
|
|
void adsp21xx_device::mac_op_mr_xop(int op)
|
|
{
|
|
INT8 shift = ((m_mstat & MSTAT_INTEGER) >> 4) ^ 1;
|
|
INT32 xop = (op >> 8) & 7;
|
|
INT32 temp;
|
|
INT64 res;
|
|
|
|
switch (op & (15<<13)) /*JB*/
|
|
{
|
|
case 0x00<<13:
|
|
/* no-op */
|
|
return;
|
|
case 0x01<<13:
|
|
/* X * Y (RND) */
|
|
xop = MAC_GETXREG_SIGNED(xop);
|
|
temp = (xop * xop) << shift;
|
|
res = (INT64)temp;
|
|
#if 0
|
|
if ((res & 0xffff) == 0x8000) res &= ~((UINT64)0x10000);
|
|
else res += (res & 0x8000) << 1;
|
|
#else
|
|
temp &= 0xffff;
|
|
res += 0x8000;
|
|
if ( temp == 0x8000 )
|
|
res &= ~((UINT64)0x10000);
|
|
#endif
|
|
break;
|
|
case 0x02<<13:
|
|
/* MR + X * Y (RND) */
|
|
xop = MAC_GETXREG_SIGNED(xop);
|
|
temp = (xop * xop) << shift;
|
|
res = m_core.mr.mr + (INT64)temp;
|
|
#if 0
|
|
if ((res & 0xffff) == 0x8000) res &= ~((UINT64)0x10000);
|
|
else res += (res & 0x8000) << 1;
|
|
#else
|
|
temp &= 0xffff;
|
|
res += 0x8000;
|
|
if ( temp == 0x8000 )
|
|
res &= ~((UINT64)0x10000);
|
|
#endif
|
|
break;
|
|
case 0x03<<13:
|
|
/* MR - X * Y (RND) */
|
|
xop = MAC_GETXREG_SIGNED(xop);
|
|
temp = (xop * xop) << shift;
|
|
res = m_core.mr.mr - (INT64)temp;
|
|
#if 0
|
|
if ((res & 0xffff) == 0x8000) res &= ~((UINT64)0x10000);
|
|
else res += (res & 0x8000) << 1;
|
|
#else
|
|
temp &= 0xffff;
|
|
res += 0x8000;
|
|
if ( temp == 0x8000 )
|
|
res &= ~((UINT64)0x10000);
|
|
#endif
|
|
break;
|
|
case 0x04<<13:
|
|
/* X * Y (SS) Clear when y = 0 */
|
|
xop = MAC_GETXREG_SIGNED(xop);
|
|
temp = (xop * xop) << shift;
|
|
res = (INT64)temp;
|
|
break;
|
|
case 0x05<<13:
|
|
/* X * Y (SU) */
|
|
xop = MAC_GETXREG_SIGNED(xop);
|
|
temp = (xop * xop) << shift;
|
|
res = (INT64)temp;
|
|
break;
|
|
case 0x06<<13:
|
|
/* X * Y (US) */
|
|
xop = MAC_GETXREG_UNSIGNED(xop);
|
|
temp = (xop * xop) << shift;
|
|
res = (INT64)temp;
|
|
break;
|
|
case 0x07<<13:
|
|
/* X * Y (UU) */
|
|
xop = MAC_GETXREG_UNSIGNED(xop);
|
|
temp = (xop * xop) << shift;
|
|
res = (INT64)temp;
|
|
break;
|
|
case 0x08<<13:
|
|
/* MR + X * Y (SS) */
|
|
xop = MAC_GETXREG_SIGNED(xop);
|
|
temp = (xop * xop) << shift;
|
|
res = m_core.mr.mr + (INT64)temp;
|
|
break;
|
|
case 0x09<<13:
|
|
/* MR + X * Y (SU) */
|
|
xop = MAC_GETXREG_SIGNED(xop);
|
|
temp = (xop * xop) << shift;
|
|
res = m_core.mr.mr + (INT64)temp;
|
|
break;
|
|
case 0x0a<<13:
|
|
/* MR + X * Y (US) */
|
|
xop = MAC_GETXREG_UNSIGNED(xop);
|
|
temp = (xop * xop) << shift;
|
|
res = m_core.mr.mr + (INT64)temp;
|
|
break;
|
|
case 0x0b<<13:
|
|
/* MR + X * Y (UU) */
|
|
xop = MAC_GETXREG_UNSIGNED(xop);
|
|
temp = (xop * xop) << shift;
|
|
res = m_core.mr.mr + (INT64)temp;
|
|
break;
|
|
case 0x0c<<13:
|
|
/* MR - X * Y (SS) */
|
|
xop = MAC_GETXREG_SIGNED(xop);
|
|
temp = (xop * xop) << shift;
|
|
res = m_core.mr.mr - (INT64)temp;
|
|
break;
|
|
case 0x0d<<13:
|
|
/* MR - X * Y (SU) */
|
|
xop = MAC_GETXREG_SIGNED(xop);
|
|
temp = (xop * xop) << shift;
|
|
res = m_core.mr.mr - (INT64)temp;
|
|
break;
|
|
case 0x0e<<13:
|
|
/* MR - X * Y (US) */
|
|
xop = MAC_GETXREG_UNSIGNED(xop);
|
|
temp = (xop * xop) << shift;
|
|
res = m_core.mr.mr - (INT64)temp;
|
|
break;
|
|
case 0x0f<<13:
|
|
/* MR - X * Y (UU) */
|
|
xop = MAC_GETXREG_UNSIGNED(xop);
|
|
temp = (xop * xop) << shift;
|
|
res = m_core.mr.mr - (INT64)temp;
|
|
break;
|
|
default:
|
|
res = 0; /* just to keep the compiler happy */
|
|
break;
|
|
}
|
|
|
|
/* set the final value */
|
|
temp = (res >> 31) & 0x1ff;
|
|
CLR_MV;
|
|
if (temp != 0x000 && temp != 0x1ff) SET_MV;
|
|
m_core.mr.mr = res;
|
|
}
|
|
|
|
|
|
|
|
/*===========================================================================
|
|
MAC operations (result in MF)
|
|
===========================================================================*/
|
|
|
|
void adsp21xx_device::mac_op_mf(int op)
|
|
{
|
|
INT8 shift = ((m_mstat & MSTAT_INTEGER) >> 4) ^ 1;
|
|
INT32 xop = (op >> 8) & 7;
|
|
INT32 yop = (op >> 11) & 3;
|
|
INT32 temp;
|
|
INT64 res;
|
|
|
|
switch (op & (15<<13)) /*JB*/
|
|
{
|
|
case 0x00<<13:
|
|
/* no-op */
|
|
return;
|
|
case 0x01<<13:
|
|
/* X * Y (RND) */
|
|
xop = MAC_GETXREG_SIGNED(xop);
|
|
yop = MAC_GETYREG_SIGNED(yop);
|
|
temp = (xop * yop) << shift;
|
|
res = (INT64)temp;
|
|
#if 0
|
|
if ((res & 0xffff) == 0x8000) res &= ~((UINT64)0x10000);
|
|
else res += (res & 0x8000) << 1;
|
|
#else
|
|
temp &= 0xffff;
|
|
res += 0x8000;
|
|
if ( temp == 0x8000 )
|
|
res &= ~((UINT64)0x10000);
|
|
#endif
|
|
break;
|
|
case 0x02<<13:
|
|
/* MR + X * Y (RND) */
|
|
xop = MAC_GETXREG_SIGNED(xop);
|
|
yop = MAC_GETYREG_SIGNED(yop);
|
|
temp = (xop * yop) << shift;
|
|
res = m_core.mr.mr + (INT64)temp;
|
|
#if 0
|
|
if ((res & 0xffff) == 0x8000) res &= ~((UINT64)0x10000);
|
|
else res += (res & 0x8000) << 1;
|
|
#else
|
|
temp &= 0xffff;
|
|
res += 0x8000;
|
|
if ( temp == 0x8000 )
|
|
res &= ~((UINT64)0x10000);
|
|
#endif
|
|
break;
|
|
case 0x03<<13:
|
|
/* MR - X * Y (RND) */
|
|
xop = MAC_GETXREG_SIGNED(xop);
|
|
yop = MAC_GETYREG_SIGNED(yop);
|
|
temp = (xop * yop) << shift;
|
|
res = m_core.mr.mr - (INT64)temp;
|
|
#if 0
|
|
if ((res & 0xffff) == 0x8000) res &= ~((UINT64)0x10000);
|
|
else res += (res & 0x8000) << 1;
|
|
#else
|
|
temp &= 0xffff;
|
|
res += 0x8000;
|
|
if ( temp == 0x8000 )
|
|
res &= ~((UINT64)0x10000);
|
|
#endif
|
|
break;
|
|
case 0x04<<13:
|
|
/* X * Y (SS) Clear when y = 0 */
|
|
xop = MAC_GETXREG_SIGNED(xop);
|
|
yop = MAC_GETYREG_SIGNED(yop);
|
|
temp = (xop * yop) << shift;
|
|
res = (INT64)temp;
|
|
break;
|
|
case 0x05<<13:
|
|
/* X * Y (SU) */
|
|
xop = MAC_GETXREG_SIGNED(xop);
|
|
yop = MAC_GETYREG_UNSIGNED(yop);
|
|
temp = (xop * yop) << shift;
|
|
res = (INT64)temp;
|
|
break;
|
|
case 0x06<<13:
|
|
/* X * Y (US) */
|
|
xop = MAC_GETXREG_UNSIGNED(xop);
|
|
yop = MAC_GETYREG_SIGNED(yop);
|
|
temp = (xop * yop) << shift;
|
|
res = (INT64)temp;
|
|
break;
|
|
case 0x07<<13:
|
|
/* X * Y (UU) */
|
|
xop = MAC_GETXREG_UNSIGNED(xop);
|
|
yop = MAC_GETYREG_UNSIGNED(yop);
|
|
temp = (xop * yop) << shift;
|
|
res = (INT64)temp;
|
|
break;
|
|
case 0x08<<13:
|
|
/* MR + X * Y (SS) */
|
|
xop = MAC_GETXREG_SIGNED(xop);
|
|
yop = MAC_GETYREG_SIGNED(yop);
|
|
temp = (xop * yop) << shift;
|
|
res = m_core.mr.mr + (INT64)temp;
|
|
break;
|
|
case 0x09<<13:
|
|
/* MR + X * Y (SU) */
|
|
xop = MAC_GETXREG_SIGNED(xop);
|
|
yop = MAC_GETYREG_UNSIGNED(yop);
|
|
temp = (xop * yop) << shift;
|
|
res = m_core.mr.mr + (INT64)temp;
|
|
break;
|
|
case 0x0a<<13:
|
|
/* MR + X * Y (US) */
|
|
xop = MAC_GETXREG_UNSIGNED(xop);
|
|
yop = MAC_GETYREG_SIGNED(yop);
|
|
temp = (xop * yop) << shift;
|
|
res = m_core.mr.mr + (INT64)temp;
|
|
break;
|
|
case 0x0b<<13:
|
|
/* MR + X * Y (UU) */
|
|
xop = MAC_GETXREG_UNSIGNED(xop);
|
|
yop = MAC_GETYREG_UNSIGNED(yop);
|
|
temp = (xop * yop) << shift;
|
|
res = m_core.mr.mr + (INT64)temp;
|
|
break;
|
|
case 0x0c<<13:
|
|
/* MR - X * Y (SS) */
|
|
xop = MAC_GETXREG_SIGNED(xop);
|
|
yop = MAC_GETYREG_SIGNED(yop);
|
|
temp = (xop * yop) << shift;
|
|
res = m_core.mr.mr - (INT64)temp;
|
|
break;
|
|
case 0x0d<<13:
|
|
/* MR - X * Y (SU) */
|
|
xop = MAC_GETXREG_SIGNED(xop);
|
|
yop = MAC_GETYREG_UNSIGNED(yop);
|
|
temp = (xop * yop) << shift;
|
|
res = m_core.mr.mr - (INT64)temp;
|
|
break;
|
|
case 0x0e<<13:
|
|
/* MR - X * Y (US) */
|
|
xop = MAC_GETXREG_UNSIGNED(xop);
|
|
yop = MAC_GETYREG_SIGNED(yop);
|
|
temp = (xop * yop) << shift;
|
|
res = m_core.mr.mr - (INT64)temp;
|
|
break;
|
|
case 0x0f<<13:
|
|
/* MR - X * Y (UU) */
|
|
xop = MAC_GETXREG_UNSIGNED(xop);
|
|
yop = MAC_GETYREG_UNSIGNED(yop);
|
|
temp = (xop * yop) << shift;
|
|
res = m_core.mr.mr - (INT64)temp;
|
|
break;
|
|
default:
|
|
res = 0; /* just to keep the compiler happy */
|
|
break;
|
|
}
|
|
|
|
/* set the final value */
|
|
m_core.mf.u = (UINT32)res >> 16;
|
|
}
|
|
|
|
|
|
|
|
/*===========================================================================
|
|
MAC operations (result in MF, yop == xop)
|
|
===========================================================================*/
|
|
|
|
void adsp21xx_device::mac_op_mf_xop(int op)
|
|
{
|
|
INT8 shift = ((m_mstat & MSTAT_INTEGER) >> 4) ^ 1;
|
|
INT32 xop = (op >> 8) & 7;
|
|
INT32 temp;
|
|
INT64 res;
|
|
|
|
switch (op & (15<<13)) /*JB*/
|
|
{
|
|
case 0x00<<13:
|
|
/* no-op */
|
|
return;
|
|
case 0x01<<13:
|
|
/* X * Y (RND) */
|
|
xop = MAC_GETXREG_SIGNED(xop);
|
|
temp = (xop * xop) << shift;
|
|
res = (INT64)temp;
|
|
#if 0
|
|
if ((res & 0xffff) == 0x8000) res &= ~((UINT64)0x10000);
|
|
else res += (res & 0x8000) << 1;
|
|
#else
|
|
temp &= 0xffff;
|
|
res += 0x8000;
|
|
if ( temp == 0x8000 )
|
|
res &= ~((UINT64)0x10000);
|
|
#endif
|
|
break;
|
|
case 0x02<<13:
|
|
/* MR + X * Y (RND) */
|
|
xop = MAC_GETXREG_SIGNED(xop);
|
|
temp = (xop * xop) << shift;
|
|
res = m_core.mr.mr + (INT64)temp;
|
|
#if 0
|
|
if ((res & 0xffff) == 0x8000) res &= ~((UINT64)0x10000);
|
|
else res += (res & 0x8000) << 1;
|
|
#else
|
|
temp &= 0xffff;
|
|
res += 0x8000;
|
|
if ( temp == 0x8000 )
|
|
res &= ~((UINT64)0x10000);
|
|
#endif
|
|
break;
|
|
case 0x03<<13:
|
|
/* MR - X * Y (RND) */
|
|
xop = MAC_GETXREG_SIGNED(xop);
|
|
temp = (xop * xop) << shift;
|
|
res = m_core.mr.mr - (INT64)temp;
|
|
#if 0
|
|
if ((res & 0xffff) == 0x8000) res &= ~((UINT64)0x10000);
|
|
else res += (res & 0x8000) << 1;
|
|
#else
|
|
temp &= 0xffff;
|
|
res += 0x8000;
|
|
if ( temp == 0x8000 )
|
|
res &= ~((UINT64)0x10000);
|
|
#endif
|
|
break;
|
|
case 0x04<<13:
|
|
/* X * Y (SS) Clear when y = 0 */
|
|
xop = MAC_GETXREG_SIGNED(xop);
|
|
temp = (xop * xop) << shift;
|
|
res = (INT64)temp;
|
|
break;
|
|
case 0x05<<13:
|
|
/* X * Y (SU) */
|
|
xop = MAC_GETXREG_SIGNED(xop);
|
|
temp = (xop * xop) << shift;
|
|
res = (INT64)temp;
|
|
break;
|
|
case 0x06<<13:
|
|
/* X * Y (US) */
|
|
xop = MAC_GETXREG_UNSIGNED(xop);
|
|
temp = (xop * xop) << shift;
|
|
res = (INT64)temp;
|
|
break;
|
|
case 0x07<<13:
|
|
/* X * Y (UU) */
|
|
xop = MAC_GETXREG_UNSIGNED(xop);
|
|
temp = (xop * xop) << shift;
|
|
res = (INT64)temp;
|
|
break;
|
|
case 0x08<<13:
|
|
/* MR + X * Y (SS) */
|
|
xop = MAC_GETXREG_SIGNED(xop);
|
|
temp = (xop * xop) << shift;
|
|
res = m_core.mr.mr + (INT64)temp;
|
|
break;
|
|
case 0x09<<13:
|
|
/* MR + X * Y (SU) */
|
|
xop = MAC_GETXREG_SIGNED(xop);
|
|
temp = (xop * xop) << shift;
|
|
res = m_core.mr.mr + (INT64)temp;
|
|
break;
|
|
case 0x0a<<13:
|
|
/* MR + X * Y (US) */
|
|
xop = MAC_GETXREG_UNSIGNED(xop);
|
|
temp = (xop * xop) << shift;
|
|
res = m_core.mr.mr + (INT64)temp;
|
|
break;
|
|
case 0x0b<<13:
|
|
/* MR + X * Y (UU) */
|
|
xop = MAC_GETXREG_UNSIGNED(xop);
|
|
temp = (xop * xop) << shift;
|
|
res = m_core.mr.mr + (INT64)temp;
|
|
break;
|
|
case 0x0c<<13:
|
|
/* MR - X * Y (SS) */
|
|
xop = MAC_GETXREG_SIGNED(xop);
|
|
temp = (xop * xop) << shift;
|
|
res = m_core.mr.mr - (INT64)temp;
|
|
break;
|
|
case 0x0d<<13:
|
|
/* MR - X * Y (SU) */
|
|
xop = MAC_GETXREG_SIGNED(xop);
|
|
temp = (xop * xop) << shift;
|
|
res = m_core.mr.mr - (INT64)temp;
|
|
break;
|
|
case 0x0e<<13:
|
|
/* MR - X * Y (US) */
|
|
xop = MAC_GETXREG_UNSIGNED(xop);
|
|
temp = (xop * xop) << shift;
|
|
res = m_core.mr.mr - (INT64)temp;
|
|
break;
|
|
case 0x0f<<13:
|
|
/* MR - X * Y (UU) */
|
|
xop = MAC_GETXREG_UNSIGNED(xop);
|
|
temp = (xop * xop) << shift;
|
|
res = m_core.mr.mr - (INT64)temp;
|
|
break;
|
|
default:
|
|
res = 0; /* just to keep the compiler happy */
|
|
break;
|
|
}
|
|
|
|
/* set the final value */
|
|
m_core.mf.u = (UINT32)res >> 16;
|
|
}
|
|
|
|
|
|
|
|
/*===========================================================================
|
|
SHIFT operations (result in SR/SE/SB)
|
|
===========================================================================*/
|
|
|
|
void adsp21xx_device::shift_op(int op)
|
|
{
|
|
INT8 sc = m_core.se.s;
|
|
INT32 xop = (op >> 8) & 7;
|
|
UINT32 res;
|
|
|
|
switch (op & (15<<11)) /*JB*/
|
|
{
|
|
case 0x00<<11:
|
|
/* LSHIFT (HI) */
|
|
xop = SHIFT_GETXREG_UNSIGNED(xop) << 16;
|
|
if (sc > 0) res = (sc < 32) ? (xop << sc) : 0;
|
|
else res = (sc > -32) ? ((UINT32)xop >> -sc) : 0;
|
|
m_core.sr.sr = res;
|
|
break;
|
|
case 0x01<<11:
|
|
/* LSHIFT (HI, OR) */
|
|
xop = SHIFT_GETXREG_UNSIGNED(xop) << 16;
|
|
if (sc > 0) res = (sc < 32) ? (xop << sc) : 0;
|
|
else res = (sc > -32) ? ((UINT32)xop >> -sc) : 0;
|
|
m_core.sr.sr |= res;
|
|
break;
|
|
case 0x02<<11:
|
|
/* LSHIFT (LO) */
|
|
xop = SHIFT_GETXREG_UNSIGNED(xop);
|
|
if (sc > 0) res = (sc < 32) ? (xop << sc) : 0;
|
|
else res = (sc > -32) ? (xop >> -sc) : 0;
|
|
m_core.sr.sr = res;
|
|
break;
|
|
case 0x03<<11:
|
|
/* LSHIFT (LO, OR) */
|
|
xop = SHIFT_GETXREG_UNSIGNED(xop);
|
|
if (sc > 0) res = (sc < 32) ? (xop << sc) : 0;
|
|
else res = (sc > -32) ? (xop >> -sc) : 0;
|
|
m_core.sr.sr |= res;
|
|
break;
|
|
case 0x04<<11:
|
|
/* ASHIFT (HI) */
|
|
xop = SHIFT_GETXREG_SIGNED(xop) << 16;
|
|
if (sc > 0) res = (sc < 32) ? (xop << sc) : 0;
|
|
else res = (sc > -32) ? (xop >> -sc) : (xop >> 31);
|
|
m_core.sr.sr = res;
|
|
break;
|
|
case 0x05<<11:
|
|
/* ASHIFT (HI, OR) */
|
|
xop = SHIFT_GETXREG_SIGNED(xop) << 16;
|
|
if (sc > 0) res = (sc < 32) ? (xop << sc) : 0;
|
|
else res = (sc > -32) ? (xop >> -sc) : (xop >> 31);
|
|
m_core.sr.sr |= res;
|
|
break;
|
|
case 0x06<<11:
|
|
/* ASHIFT (LO) */
|
|
xop = SHIFT_GETXREG_SIGNED(xop);
|
|
if (sc > 0) res = (sc < 32) ? (xop << sc) : 0;
|
|
else res = (sc > -32) ? (xop >> -sc) : (xop >> 31);
|
|
m_core.sr.sr = res;
|
|
break;
|
|
case 0x07<<11:
|
|
/* ASHIFT (LO, OR) */
|
|
xop = SHIFT_GETXREG_SIGNED(xop);
|
|
if (sc > 0) res = (sc < 32) ? (xop << sc) : 0;
|
|
else res = (sc > -32) ? (xop >> -sc) : (xop >> 31);
|
|
m_core.sr.sr |= res;
|
|
break;
|
|
case 0x08<<11:
|
|
/* NORM (HI) */
|
|
xop = SHIFT_GETXREG_SIGNED(xop) << 16;
|
|
if (sc > 0)
|
|
{
|
|
xop = ((UINT32)xop >> 1) | ((m_astat & CFLAG) << 28);
|
|
res = xop >> (sc - 1);
|
|
}
|
|
else res = (sc > -32) ? (xop << -sc) : 0;
|
|
m_core.sr.sr = res;
|
|
break;
|
|
case 0x09<<11:
|
|
/* NORM (HI, OR) */
|
|
xop = SHIFT_GETXREG_SIGNED(xop) << 16;
|
|
if (sc > 0)
|
|
{
|
|
xop = ((UINT32)xop >> 1) | ((m_astat & CFLAG) << 28);
|
|
res = xop >> (sc - 1);
|
|
}
|
|
else res = (sc > -32) ? (xop << -sc) : 0;
|
|
m_core.sr.sr |= res;
|
|
break;
|
|
case 0x0a<<11:
|
|
/* NORM (LO) */
|
|
xop = SHIFT_GETXREG_UNSIGNED(xop);
|
|
if (sc > 0) res = (sc < 32) ? (xop >> sc) : 0;
|
|
else res = (sc > -32) ? (xop << -sc) : 0;
|
|
m_core.sr.sr = res;
|
|
break;
|
|
case 0x0b<<11:
|
|
/* NORM (LO, OR) */
|
|
xop = SHIFT_GETXREG_UNSIGNED(xop);
|
|
if (sc > 0) res = (sc < 32) ? (xop >> sc) : 0;
|
|
else res = (sc > -32) ? (xop << -sc) : 0;
|
|
m_core.sr.sr |= res;
|
|
break;
|
|
case 0x0c<<11:
|
|
/* EXP (HI) */
|
|
xop = SHIFT_GETXREG_SIGNED(xop) << 16;
|
|
res = 0;
|
|
if (xop < 0)
|
|
{
|
|
SET_SS;
|
|
while ((xop & 0x40000000) != 0) res++, xop <<= 1;
|
|
}
|
|
else
|
|
{
|
|
CLR_SS;
|
|
xop |= 0x8000;
|
|
while ((xop & 0x40000000) == 0) res++, xop <<= 1;
|
|
}
|
|
m_core.se.s = -res;
|
|
break;
|
|
case 0x0d<<11:
|
|
/* EXP (HIX) */
|
|
xop = SHIFT_GETXREG_SIGNED(xop) << 16;
|
|
if (GET_V)
|
|
{
|
|
m_core.se.s = 1;
|
|
if (xop < 0) CLR_SS;
|
|
else SET_SS;
|
|
}
|
|
else
|
|
{
|
|
res = 0;
|
|
if (xop < 0)
|
|
{
|
|
SET_SS;
|
|
while ((xop & 0x40000000) != 0) res++, xop <<= 1;
|
|
}
|
|
else
|
|
{
|
|
CLR_SS;
|
|
xop |= 0x8000;
|
|
while ((xop & 0x40000000) == 0) res++, xop <<= 1;
|
|
}
|
|
m_core.se.s = -res;
|
|
}
|
|
break;
|
|
case 0x0e<<11:
|
|
/* EXP (LO) */
|
|
if (m_core.se.s == -15)
|
|
{
|
|
xop = SHIFT_GETXREG_SIGNED(xop);
|
|
res = 15;
|
|
if (GET_SS)
|
|
while ((xop & 0x8000) != 0) res++, xop <<= 1;
|
|
else
|
|
{
|
|
xop = (xop << 1) | 1;
|
|
while ((xop & 0x10000) == 0) res++, xop <<= 1;
|
|
}
|
|
m_core.se.s = -res;
|
|
}
|
|
break;
|
|
case 0x0f<<11:
|
|
/* EXPADJ */
|
|
xop = SHIFT_GETXREG_SIGNED(xop) << 16;
|
|
res = 0;
|
|
if (xop < 0)
|
|
while ((xop & 0x40000000) != 0) res++, xop <<= 1;
|
|
else
|
|
{
|
|
xop |= 0x8000;
|
|
while ((xop & 0x40000000) == 0) res++, xop <<= 1;
|
|
}
|
|
if (res < -m_core.sb.s)
|
|
m_core.sb.s = -res;
|
|
break;
|
|
}
|
|
}
|
|
|
|
|
|
|
|
/*===========================================================================
|
|
Immediate SHIFT operations (result in SR/SE/SB)
|
|
===========================================================================*/
|
|
|
|
void adsp21xx_device::shift_op_imm(int op)
|
|
{
|
|
INT8 sc = (INT8)op;
|
|
INT32 xop = (op >> 8) & 7;
|
|
UINT32 res;
|
|
|
|
switch (op & (15<<11)) /*JB*/
|
|
{
|
|
case 0x00<<11:
|
|
/* LSHIFT (HI) */
|
|
xop = SHIFT_GETXREG_UNSIGNED(xop) << 16;
|
|
if (sc > 0) res = (sc < 32) ? (xop << sc) : 0;
|
|
else res = (sc > -32) ? ((UINT32)xop >> -sc) : 0;
|
|
m_core.sr.sr = res;
|
|
break;
|
|
case 0x01<<11:
|
|
/* LSHIFT (HI, OR) */
|
|
xop = SHIFT_GETXREG_UNSIGNED(xop) << 16;
|
|
if (sc > 0) res = (sc < 32) ? (xop << sc) : 0;
|
|
else res = (sc > -32) ? ((UINT32)xop >> -sc) : 0;
|
|
m_core.sr.sr |= res;
|
|
break;
|
|
case 0x02<<11:
|
|
/* LSHIFT (LO) */
|
|
xop = SHIFT_GETXREG_UNSIGNED(xop);
|
|
if (sc > 0) res = (sc < 32) ? (xop << sc) : 0;
|
|
else res = (sc > -32) ? (xop >> -sc) : 0;
|
|
m_core.sr.sr = res;
|
|
break;
|
|
case 0x03<<11:
|
|
/* LSHIFT (LO, OR) */
|
|
xop = SHIFT_GETXREG_UNSIGNED(xop);
|
|
if (sc > 0) res = (sc < 32) ? (xop << sc) : 0;
|
|
else res = (sc > -32) ? (xop >> -sc) : 0;
|
|
m_core.sr.sr |= res;
|
|
break;
|
|
case 0x04<<11:
|
|
/* ASHIFT (HI) */
|
|
xop = SHIFT_GETXREG_SIGNED(xop) << 16;
|
|
if (sc > 0) res = (sc < 32) ? (xop << sc) : 0;
|
|
else res = (sc > -32) ? (xop >> -sc) : (xop >> 31);
|
|
m_core.sr.sr = res;
|
|
break;
|
|
case 0x05<<11:
|
|
/* ASHIFT (HI, OR) */
|
|
xop = SHIFT_GETXREG_SIGNED(xop) << 16;
|
|
if (sc > 0) res = (sc < 32) ? (xop << sc) : 0;
|
|
else res = (sc > -32) ? (xop >> -sc) : (xop >> 31);
|
|
m_core.sr.sr |= res;
|
|
break;
|
|
case 0x06<<11:
|
|
/* ASHIFT (LO) */
|
|
xop = SHIFT_GETXREG_SIGNED(xop);
|
|
if (sc > 0) res = (sc < 32) ? (xop << sc) : 0;
|
|
else res = (sc > -32) ? (xop >> -sc) : (xop >> 31);
|
|
m_core.sr.sr = res;
|
|
break;
|
|
case 0x07<<11:
|
|
/* ASHIFT (LO, OR) */
|
|
xop = SHIFT_GETXREG_SIGNED(xop);
|
|
if (sc > 0) res = (sc < 32) ? (xop << sc) : 0;
|
|
else res = (sc > -32) ? (xop >> -sc) : (xop >> 31);
|
|
m_core.sr.sr |= res;
|
|
break;
|
|
case 0x08<<11:
|
|
/* NORM (HI) */
|
|
xop = SHIFT_GETXREG_SIGNED(xop) << 16;
|
|
if (sc > 0)
|
|
{
|
|
xop = ((UINT32)xop >> 1) | ((m_astat & CFLAG) << 28);
|
|
res = xop >> (sc - 1);
|
|
}
|
|
else res = (sc > -32) ? (xop << -sc) : 0;
|
|
m_core.sr.sr = res;
|
|
break;
|
|
case 0x09<<11:
|
|
/* NORM (HI, OR) */
|
|
xop = SHIFT_GETXREG_SIGNED(xop) << 16;
|
|
if (sc > 0)
|
|
{
|
|
xop = ((UINT32)xop >> 1) | ((m_astat & CFLAG) << 28);
|
|
res = xop >> (sc - 1);
|
|
}
|
|
else res = (sc > -32) ? (xop << -sc) : 0;
|
|
m_core.sr.sr |= res;
|
|
break;
|
|
case 0x0a<<11:
|
|
/* NORM (LO) */
|
|
xop = SHIFT_GETXREG_UNSIGNED(xop);
|
|
if (sc > 0) res = (sc < 32) ? (xop >> sc) : 0;
|
|
else res = (sc > -32) ? (xop << -sc) : 0;
|
|
m_core.sr.sr = res;
|
|
break;
|
|
case 0x0b<<11:
|
|
/* NORM (LO, OR) */
|
|
xop = SHIFT_GETXREG_UNSIGNED(xop);
|
|
if (sc > 0) res = (sc < 32) ? (xop >> sc) : 0;
|
|
else res = (sc > -32) ? (xop << -sc) : 0;
|
|
m_core.sr.sr |= res;
|
|
break;
|
|
}
|
|
}
|