mirror of
https://github.com/holub/mame
synced 2025-04-25 17:56:43 +03:00
nds: Added timers, interrupt management, partial DMA, and ARM7 halt-until-IRQ. [R. Belmont]
This commit is contained in:
parent
2d1ca63a21
commit
2bb61a2553
@ -1,10 +1,16 @@
|
||||
// license:BSD-3-Clause
|
||||
// copyright-holders:Ryan Holtz
|
||||
// copyright-holders:Ryan Holtz, R. Belmont
|
||||
/***************************************************************************
|
||||
|
||||
nds.cpp
|
||||
|
||||
Skeleton driver for first-generation Nintendo DS.
|
||||
Preliminary driver for first-generation Nintendo DS.
|
||||
|
||||
Tech info: http://problemkaputt.de/gbatek.htm
|
||||
|
||||
Notes:
|
||||
Timers and DMAs 0-3 are ARM9's, 4-7 are ARM7's.
|
||||
Interrupt registers [0] is ARM9, [1] is ARM7.
|
||||
|
||||
***************************************************************************/
|
||||
|
||||
@ -13,6 +19,44 @@
|
||||
|
||||
#define VERBOSE_LEVEL (0)
|
||||
|
||||
// Measured value from GBATEK. Actual crystal unknown.
|
||||
#define MASTER_CLOCK (33513982)
|
||||
|
||||
#define INT_VBL 0x00000001
|
||||
#define INT_HBL 0x00000002
|
||||
#define INT_VCNT 0x00000004
|
||||
#define INT_TM0_OVERFLOW 0x00000008
|
||||
#define INT_TM1_OVERFLOW 0x00000010
|
||||
#define INT_TM2_OVERFLOW 0x00000020
|
||||
#define INT_TM3_OVERFLOW 0x00000040
|
||||
#define INT_SIO 0x00000080 // also RCNT/RTC (arm7 only)
|
||||
#define INT_DMA0 0x00000100
|
||||
#define INT_DMA1 0x00000200
|
||||
#define INT_DMA2 0x00000400
|
||||
#define INT_DMA3 0x00000800
|
||||
#define INT_KEYPAD 0x00001000
|
||||
#define INT_GAMEPAK 0x00002000 // GBA slot IRQ line (never used?)
|
||||
#define INT_NA1 0x00004000 // unused
|
||||
#define INT_NA2 0x00008000 // unused
|
||||
#define INT_IPCSYNC 0x00010000
|
||||
#define INT_IPCSENDEMPTY 0x00020000
|
||||
#define INT_IPCRECVNOTEMPTY 0x00040000
|
||||
#define INT_CARDXFERCOMPLETE 0x00080000
|
||||
#define INT_CARDIREQ 0x00100000
|
||||
#define INT_GEOCMDFIFO 0x00200000 // arm9 only
|
||||
#define INT_SCREENUNFOLD 0x00400000 // arm7 only
|
||||
#define INT_SPIBUS 0x00800000 // arm7 only
|
||||
#define INT_WIFI 0x01000000 // arm7 only - also DSP on DSi
|
||||
#define INT_CAMERA 0x02000000 // DSi only
|
||||
#define INT_NA3 0x04000000
|
||||
#define INT_NA4 0x08000000
|
||||
#define INT_NEWDMA0 0x10000000 // DSi only
|
||||
#define INT_NEWDMA1 0x20000000 // DSi only
|
||||
#define INT_NEWDMA2 0x40000000 // DSi only
|
||||
#define INT_NEWDMA3 0x80000000 // DSi only
|
||||
|
||||
static const uint32_t timer_clks[4] = { MASTER_CLOCK, MASTER_CLOCK / 64, MASTER_CLOCK / 256, MASTER_CLOCK / 1024 };
|
||||
|
||||
static inline void ATTR_PRINTF(3,4) verboselog(device_t &device, int n_level, const char *s_fmt, ...)
|
||||
{
|
||||
if( VERBOSE_LEVEL >= n_level )
|
||||
@ -31,11 +75,106 @@ READ32_MEMBER(nds_state::arm7_io_r)
|
||||
uint8_t temp1, temp2;
|
||||
switch(offset)
|
||||
{
|
||||
case TIMER_OFFSET:
|
||||
case TIMER_OFFSET+1:
|
||||
case TIMER_OFFSET+2:
|
||||
case TIMER_OFFSET+3:
|
||||
{
|
||||
uint32_t elapsed;
|
||||
double time, ticks;
|
||||
int timer = (offset - TIMER_OFFSET) + 4;
|
||||
|
||||
printf("Read timer reg %x (PC=%x)\n", timer, space.device().safe_pc());
|
||||
|
||||
// update times for
|
||||
if (m_timer_regs[timer] & 0x800000)
|
||||
{
|
||||
if (m_timer_regs[timer] & 0x00040000)
|
||||
{
|
||||
elapsed = m_timer_regs[timer] & 0xffff;
|
||||
}
|
||||
else
|
||||
{
|
||||
time = 0.1; //m_tmr_timer[timer]->elapsed().as_double();
|
||||
|
||||
ticks = (double)(0x10000 - (m_timer_regs[timer] & 0xffff));
|
||||
|
||||
// printf("time %f ticks %f 1/hz %f\n", time, ticks, 1.0 / m_timer_hz[timer]);
|
||||
|
||||
time *= ticks;
|
||||
time /= (1.0 / m_timer_hz[timer]);
|
||||
|
||||
elapsed = (uint32_t)time;
|
||||
}
|
||||
|
||||
// printf("elapsed = %x\n", elapsed);
|
||||
}
|
||||
else
|
||||
{
|
||||
// printf("Reading inactive timer!\n");
|
||||
elapsed = 0;
|
||||
}
|
||||
|
||||
return (m_timer_regs[timer] & 0xffff0000) | (elapsed & 0xffff);
|
||||
}
|
||||
break;
|
||||
|
||||
case IME_OFFSET:
|
||||
return m_ime[1];
|
||||
|
||||
case IE_OFFSET:
|
||||
return m_ie[1];
|
||||
|
||||
case IF_OFFSET:
|
||||
return m_if[1];
|
||||
|
||||
case IPCSYNC_OFFSET:
|
||||
return m_arm7_ipcsync;
|
||||
|
||||
case AUX_SPI_CNT_OFFSET:
|
||||
printf("arm7: read AUX_SPI_CNT mask %08x\n", mem_mask);
|
||||
return 0;
|
||||
break;
|
||||
|
||||
case GAMECARD_BUS_CTRL_OFFSET:
|
||||
//printf("arm7: read GAMECARD_BUS_CTRL (%08x) mask %08x\n", m_gamecard_ctrl, mem_mask);
|
||||
return m_gamecard_ctrl;
|
||||
break;
|
||||
|
||||
case GAMECARD_DATA_OFFSET:
|
||||
printf("arm7: read to GAMECARD_DATA mask %08x\n", mem_mask);
|
||||
return 0xffffffff;
|
||||
break;
|
||||
|
||||
case GAMECARD_DATA_2_OFFSET:
|
||||
printf("arm7: read to GAMECARD_DATA2 mask %08x\n", mem_mask);
|
||||
return 0xffffffff;
|
||||
break;
|
||||
|
||||
case GAMECARD_DATA_IN_OFFSET:
|
||||
//printf("arm7: read to GAMECARD_DATA_IN mask %08x (len = %x)\n", mem_mask, m_cartdata_len);
|
||||
if (m_cartdata_len >= 4)
|
||||
{
|
||||
m_cartdata_len -= 4;
|
||||
}
|
||||
else
|
||||
{
|
||||
m_cartdata_len = 0;
|
||||
}
|
||||
|
||||
if (m_cartdata_len == 0)
|
||||
{
|
||||
printf("NDS: xfer over\n");
|
||||
m_gamecard_ctrl &= ~GAMECARD_DATA_READY;
|
||||
m_gamecard_ctrl &= ~GAMECARD_BLOCK_BUSY;
|
||||
}
|
||||
return 0xffffffff;
|
||||
break;
|
||||
|
||||
case SPI_CTRL_OFFSET:
|
||||
//printf("arm7: read SPI_CTRL mask %08x\n", mem_mask);
|
||||
return 0;
|
||||
break;
|
||||
|
||||
case POSTFLG_OFFSET:
|
||||
/* Bit Use
|
||||
@ -60,14 +199,139 @@ WRITE32_MEMBER(nds_state::arm7_io_w)
|
||||
{
|
||||
switch(offset)
|
||||
{
|
||||
case TIMER_OFFSET:
|
||||
case TIMER_OFFSET+1:
|
||||
case TIMER_OFFSET+2:
|
||||
case TIMER_OFFSET+3:
|
||||
{
|
||||
double rate, clocksel;
|
||||
uint32_t old_timer_regs;
|
||||
|
||||
int timer = (offset - TIMER_OFFSET)+4;
|
||||
|
||||
old_timer_regs = m_timer_regs[timer];
|
||||
|
||||
m_timer_regs[timer] = (m_timer_regs[timer] & ~(mem_mask & 0xFFFF0000)) | (data & (mem_mask & 0xFFFF0000));
|
||||
|
||||
printf("%08x to timer %d (mask %08x PC %x)\n", data, timer, ~mem_mask, space.device().safe_pc());
|
||||
|
||||
if (ACCESSING_BITS_0_15)
|
||||
{
|
||||
m_timer_reload[timer] = ((m_timer_reload[timer] & ~mem_mask) | (data & mem_mask)) & 0x0000FFFF;
|
||||
m_timer_recalc[timer] = 1;
|
||||
}
|
||||
|
||||
// enabling this timer?
|
||||
if ((ACCESSING_BITS_16_31) && (data & 0x800000))
|
||||
{
|
||||
double final;
|
||||
|
||||
if ((old_timer_regs & 0x00800000) == 0) // start bit 0 -> 1
|
||||
{
|
||||
m_timer_regs[timer] = (m_timer_regs[timer] & 0xFFFF0000) | (m_timer_reload[timer] & 0x0000FFFF);
|
||||
}
|
||||
|
||||
rate = 0x10000 - (m_timer_regs[timer] & 0xffff);
|
||||
|
||||
clocksel = timer_clks[(m_timer_regs[timer] >> 16) & 3];
|
||||
|
||||
final = clocksel / rate;
|
||||
|
||||
m_timer_hz[timer] = final;
|
||||
|
||||
m_timer_recalc[timer] = 0;
|
||||
|
||||
printf("Enabling timer %d @ %f Hz regs %08x\n", timer, final, m_timer_regs[timer]);
|
||||
|
||||
// enable the timer
|
||||
if( !(data & 0x40000) ) // if we're not in Count-Up mode
|
||||
{
|
||||
attotime time = attotime::from_hz(final);
|
||||
m_tmr_timer[timer]->adjust(time, timer, time);
|
||||
}
|
||||
}
|
||||
}
|
||||
break;
|
||||
|
||||
case IME_OFFSET:
|
||||
printf("ARM7: %08x to IME\n", data);
|
||||
COMBINE_DATA(&m_ime[1]);
|
||||
break;
|
||||
|
||||
case IE_OFFSET:
|
||||
printf("ARM7: %08x to IE\n", data);
|
||||
COMBINE_DATA(&m_ie[1]);
|
||||
break;
|
||||
|
||||
case IF_OFFSET:
|
||||
COMBINE_DATA(&m_if[1]);
|
||||
break;
|
||||
|
||||
case IPCSYNC_OFFSET:
|
||||
printf("ARM7: %x to IPCSYNC\n", data);
|
||||
//printf("ARM7: %x to IPCSYNC\n", data);
|
||||
m_arm9_ipcsync &= ~0xf;
|
||||
m_arm9_ipcsync |= ((data >> 8) & 0xf);
|
||||
m_arm7_ipcsync &= 0xf;
|
||||
m_arm7_ipcsync |= (data & ~0xf);
|
||||
break;
|
||||
|
||||
case AUX_SPI_CNT_OFFSET:
|
||||
//printf("arm7: %08x to AUX_SPI_CNT mask %08x\n", data, mem_mask);
|
||||
m_spicnt &= 0x0080;
|
||||
m_spicnt |= (data & 0xe043);
|
||||
|
||||
break;
|
||||
|
||||
case GAMECARD_BUS_CTRL_OFFSET:
|
||||
//printf("arm7: %08x to GAMECARD_BUS_CTRL mask %08x\n", data, mem_mask);
|
||||
m_gamecard_ctrl &= GAMECARD_DATA_READY;
|
||||
m_gamecard_ctrl |= (data & ~GAMECARD_DATA_READY);
|
||||
|
||||
if (!(m_spicnt & (1<<15)))
|
||||
{
|
||||
return;
|
||||
}
|
||||
|
||||
if (!(m_gamecard_ctrl & GAMECARD_BLOCK_BUSY))
|
||||
{
|
||||
return;
|
||||
}
|
||||
|
||||
m_cartdata_len = (m_gamecard_ctrl >> 24) & 7;
|
||||
if (m_cartdata_len == 7)
|
||||
{
|
||||
m_cartdata_len = 4;
|
||||
}
|
||||
else if (m_cartdata_len != 0)
|
||||
{
|
||||
m_cartdata_len = 256 << m_cartdata_len;
|
||||
}
|
||||
printf("nds: cartdata for transfer = %x\n", m_cartdata_len);
|
||||
|
||||
if (m_cartdata_len > 0)
|
||||
{
|
||||
m_gamecard_ctrl |= GAMECARD_DATA_READY;
|
||||
}
|
||||
else
|
||||
{
|
||||
printf("NDS: xfer over\n");
|
||||
m_gamecard_ctrl &= ~GAMECARD_DATA_READY;
|
||||
m_gamecard_ctrl &= ~GAMECARD_BLOCK_BUSY;
|
||||
}
|
||||
break;
|
||||
|
||||
case GAMECARD_DATA_OFFSET:
|
||||
//printf("arm7: %08x to GAMECARD_DATA mask %08x\n", data, mem_mask);
|
||||
break;
|
||||
|
||||
case GAMECARD_DATA_2_OFFSET:
|
||||
//printf("arm7: %08x to GAMECARD_DATA2 mask %08x\n", data, mem_mask);
|
||||
break;
|
||||
|
||||
case SPI_CTRL_OFFSET:
|
||||
//printf("arm7: %08x to SPI_CTRL mask %08x\n", data, mem_mask);
|
||||
break;
|
||||
|
||||
case POSTFLG_OFFSET:
|
||||
/* Bit Use
|
||||
* 0 0=Booting, 1=Booted (set by BIOS/firmware)
|
||||
@ -77,6 +341,16 @@ WRITE32_MEMBER(nds_state::arm7_io_w)
|
||||
m_arm7_postflg &= ~POSTFLG_PBF_MASK;
|
||||
m_arm7_postflg |= data & POSTFLG_PBF_MASK;
|
||||
}
|
||||
|
||||
if (ACCESSING_BITS_8_15)
|
||||
{
|
||||
if ((data>>8) & 0x80)
|
||||
{
|
||||
printf("arm7: HALT\n"); // halts the arm7 until an interrupt occurs
|
||||
m_arm7->suspend(SUSPEND_REASON_HALT, 1);
|
||||
m_arm7halted = true;
|
||||
}
|
||||
}
|
||||
break;
|
||||
default:
|
||||
verboselog(*this, 0, "[ARM7] [IO] Unknown write: %08x = %08x (%08x)\n", offset*4, data, mem_mask);
|
||||
@ -88,6 +362,59 @@ READ32_MEMBER(nds_state::arm9_io_r)
|
||||
{
|
||||
switch(offset)
|
||||
{
|
||||
case TIMER_OFFSET:
|
||||
case TIMER_OFFSET+1:
|
||||
case TIMER_OFFSET+2:
|
||||
case TIMER_OFFSET+3:
|
||||
{
|
||||
uint32_t elapsed;
|
||||
double time, ticks;
|
||||
int timer = (offset - TIMER_OFFSET);
|
||||
|
||||
//printf("Read timer reg %x (PC=%x)\n", timer, space.device().safe_pc());
|
||||
|
||||
// update times for
|
||||
if (m_timer_regs[timer] & 0x800000)
|
||||
{
|
||||
if (m_timer_regs[timer] & 0x00040000)
|
||||
{
|
||||
elapsed = m_timer_regs[timer] & 0xffff;
|
||||
}
|
||||
else
|
||||
{
|
||||
time = 0.1; //m_tmr_timer[timer]->elapsed().as_double();
|
||||
|
||||
ticks = (double)(0x10000 - (m_timer_regs[timer] & 0xffff));
|
||||
|
||||
// printf("time %f ticks %f 1/hz %f\n", time, ticks, 1.0 / m_timer_hz[timer]);
|
||||
|
||||
time *= ticks;
|
||||
time /= (1.0 / m_timer_hz[timer]);
|
||||
|
||||
elapsed = (uint32_t)time;
|
||||
}
|
||||
|
||||
// printf("elapsed = %x\n", elapsed);
|
||||
}
|
||||
else
|
||||
{
|
||||
// printf("Reading inactive timer!\n");
|
||||
elapsed = 0;
|
||||
}
|
||||
|
||||
return (m_timer_regs[timer] & 0xffff0000) | (elapsed & 0xffff);
|
||||
}
|
||||
break;
|
||||
|
||||
case IME_OFFSET:
|
||||
return m_ime[0];
|
||||
|
||||
case IE_OFFSET:
|
||||
return m_ie[0];
|
||||
|
||||
case IF_OFFSET:
|
||||
return m_if[0];
|
||||
|
||||
case IPCSYNC_OFFSET:
|
||||
return m_arm9_ipcsync;
|
||||
|
||||
@ -109,6 +436,74 @@ WRITE32_MEMBER(nds_state::arm9_io_w)
|
||||
{
|
||||
switch(offset)
|
||||
{
|
||||
case TIMER_OFFSET:
|
||||
case TIMER_OFFSET+1:
|
||||
case TIMER_OFFSET+2:
|
||||
case TIMER_OFFSET+3:
|
||||
{
|
||||
double rate, clocksel;
|
||||
uint32_t old_timer_regs;
|
||||
|
||||
int timer = (offset - TIMER_OFFSET)+4;
|
||||
|
||||
old_timer_regs = m_timer_regs[timer];
|
||||
|
||||
m_timer_regs[timer] = (m_timer_regs[timer] & ~(mem_mask & 0xFFFF0000)) | (data & (mem_mask & 0xFFFF0000));
|
||||
|
||||
printf("%x to timer %d (mask %x PC %x)\n", data, timer, ~mem_mask, space.device().safe_pc());
|
||||
|
||||
if (ACCESSING_BITS_0_15)
|
||||
{
|
||||
m_timer_reload[timer] = ((m_timer_reload[timer] & ~mem_mask) | (data & mem_mask)) & 0x0000FFFF;
|
||||
m_timer_recalc[timer] = 1;
|
||||
}
|
||||
|
||||
// enabling this timer?
|
||||
if ((ACCESSING_BITS_16_31) && (data & 0x800000))
|
||||
{
|
||||
double final;
|
||||
|
||||
if ((old_timer_regs & 0x00800000) == 0) // start bit 0 -> 1
|
||||
{
|
||||
m_timer_regs[timer] = (m_timer_regs[timer] & 0xFFFF0000) | (m_timer_reload[timer] & 0x0000FFFF);
|
||||
}
|
||||
|
||||
rate = 0x10000 - (m_timer_regs[timer] & 0xffff);
|
||||
|
||||
clocksel = timer_clks[(m_timer_regs[timer] >> 16) & 3];
|
||||
|
||||
final = clocksel / rate;
|
||||
|
||||
m_timer_hz[timer] = final;
|
||||
|
||||
m_timer_recalc[timer] = 0;
|
||||
|
||||
printf("Enabling timer %d @ %f Hz\n", timer, final);
|
||||
|
||||
// enable the timer
|
||||
if( !(data & 0x40000) ) // if we're not in Count-Up mode
|
||||
{
|
||||
attotime time = attotime::from_hz(final);
|
||||
m_tmr_timer[timer]->adjust(time, timer, time);
|
||||
}
|
||||
}
|
||||
}
|
||||
break;
|
||||
|
||||
case IME_OFFSET:
|
||||
printf("ARM9: %08x to IME\n", data);
|
||||
COMBINE_DATA(&m_ime[0]);
|
||||
break;
|
||||
|
||||
case IE_OFFSET:
|
||||
printf("ARM9: %08x to IE\n", data);
|
||||
COMBINE_DATA(&m_ie[0]);
|
||||
break;
|
||||
|
||||
case IF_OFFSET:
|
||||
COMBINE_DATA(&m_if[0]);
|
||||
break;
|
||||
|
||||
case IPCSYNC_OFFSET:
|
||||
printf("ARM9: %x to IPCSYNC\n", data);
|
||||
m_arm7_ipcsync &= ~0xf;
|
||||
@ -190,15 +585,15 @@ WRITE32_MEMBER(nds_state::arm9_io_w)
|
||||
static ADDRESS_MAP_START( nds_arm7_map, AS_PROGRAM, 32, nds_state )
|
||||
AM_RANGE(0x00000000, 0x00003fff) AM_ROM AM_REGION("arm7", 0)
|
||||
AM_RANGE(0x02000000, 0x023fffff) AM_RAM AM_MIRROR(0x00400000) AM_SHARE("mainram")
|
||||
AM_RANGE(0x03000000, 0x03007fff) AM_DEVICE("nds7wram", address_map_bank_device, amap32) AM_MIRROR(0x007f0000)
|
||||
AM_RANGE(0x03000000, 0x03007fff) AM_MIRROR(0x007f8000) AM_DEVICE("nds7wram", address_map_bank_device, amap32)
|
||||
AM_RANGE(0x03800000, 0x0380ffff) AM_RAM AM_MIRROR(0x007f0000) AM_SHARE("arm7ram")
|
||||
AM_RANGE(0x04000000, 0x0400ffff) AM_READWRITE(arm7_io_r, arm7_io_w)
|
||||
AM_RANGE(0x04000000, 0x0410ffff) AM_READWRITE(arm7_io_r, arm7_io_w)
|
||||
ADDRESS_MAP_END
|
||||
|
||||
static ADDRESS_MAP_START( nds_arm9_map, AS_PROGRAM, 32, nds_state )
|
||||
AM_RANGE(0x02000000, 0x023fffff) AM_RAM AM_MIRROR(0x00400000) AM_SHARE("mainram")
|
||||
AM_RANGE(0x03000000, 0x03007fff) AM_DEVICE("nds9wram", address_map_bank_device, amap32) AM_MIRROR(0x00ff0000)
|
||||
AM_RANGE(0x04000000, 0x0400ffff) AM_READWRITE(arm9_io_r, arm9_io_w)
|
||||
AM_RANGE(0x03000000, 0x03007fff) AM_MIRROR(0x00ff8000) AM_DEVICE("nds9wram", address_map_bank_device, amap32)
|
||||
AM_RANGE(0x04000000, 0x0410ffff) AM_READWRITE(arm9_io_r, arm9_io_w)
|
||||
AM_RANGE(0xffff0000, 0xffff0fff) AM_ROM AM_MIRROR(0x1000) AM_REGION("arm9", 0)
|
||||
ADDRESS_MAP_END
|
||||
|
||||
@ -232,7 +627,6 @@ WRITE32_MEMBER(nds_state::wram_arm7mirror_w) { COMBINE_DATA(&m_arm7ram[offset]);
|
||||
static INPUT_PORTS_START( nds )
|
||||
INPUT_PORTS_END
|
||||
|
||||
|
||||
void nds_state::machine_reset()
|
||||
{
|
||||
m_arm7_postflg = 0;
|
||||
@ -240,17 +634,343 @@ void nds_state::machine_reset()
|
||||
m_wramcnt = 0;
|
||||
m_arm7wrambnk->set_bank(0);
|
||||
m_arm9wrambnk->set_bank(0);
|
||||
m_arm7halted = false;
|
||||
}
|
||||
|
||||
void nds_state::machine_start()
|
||||
{
|
||||
int i;
|
||||
|
||||
for (i = 0; i < 8; i++)
|
||||
{
|
||||
m_dma_timer[i] = machine().scheduler().timer_alloc(timer_expired_delegate(FUNC(nds_state::dma_complete),this));
|
||||
m_dma_timer[i]->adjust(attotime::never, i);
|
||||
|
||||
m_tmr_timer[i] = machine().scheduler().timer_alloc(timer_expired_delegate(FUNC(nds_state::timer_expire),this));
|
||||
m_tmr_timer[i]->adjust(attotime::never, i);
|
||||
}
|
||||
|
||||
m_irq_timer = machine().scheduler().timer_alloc(timer_expired_delegate(FUNC(nds_state::handle_irq),this));
|
||||
m_irq_timer->adjust(attotime::never);
|
||||
}
|
||||
|
||||
TIMER_CALLBACK_MEMBER(nds_state::dma_complete)
|
||||
{
|
||||
#if 0
|
||||
static const uint32_t ch_int[8] = { INT_DMA0, INT_DMA1, INT_DMA2, INT_DMA3, INT_DMA0, INT_DMA1, INT_DMA2, INT_DMA3 };
|
||||
|
||||
uintptr_t ch = param;
|
||||
|
||||
// printf("dma complete: ch %d\n", ch);
|
||||
|
||||
m_dma_timer[ch]->adjust(attotime::never);
|
||||
|
||||
int ctrl = DMACNT_H(ch);
|
||||
|
||||
// IRQ
|
||||
if (ctrl & 0x4000)
|
||||
{
|
||||
request_irq(ch_int[ch]);
|
||||
}
|
||||
|
||||
// if we're supposed to repeat, don't clear "active" and then the next vbl/hbl will retrigger us
|
||||
// always clear active for immediate DMAs though
|
||||
if (!((ctrl>>9) & 1) || ((ctrl & 0x3000) == 0))
|
||||
{
|
||||
DMACNT_H_RESET(ch, 0x8000); // clear "active" bit
|
||||
}
|
||||
else
|
||||
{
|
||||
// if repeat, reload the count
|
||||
if ((ctrl>>9) & 1)
|
||||
{
|
||||
m_dma_cnt[ch] = DMACNT_L(ch);
|
||||
|
||||
// if increment & reload mode, reload the destination
|
||||
if (((ctrl>>5)&3) == 3)
|
||||
{
|
||||
m_dma_dst[ch] = DMADAD(ch);
|
||||
}
|
||||
}
|
||||
}
|
||||
#endif
|
||||
}
|
||||
|
||||
void nds_state::dma_exec(int ch)
|
||||
{
|
||||
#if 0
|
||||
address_space &space;
|
||||
uint32_t src = m_dma_src[ch];
|
||||
uint32_t dst = m_dma_dst[ch];
|
||||
uint16_t ctrl = DMACNT_H(ch);
|
||||
int srcadd = (ctrl >> 7) & 3;
|
||||
int dstadd = (ctrl >> 5) & 3;
|
||||
|
||||
if (ch > 4)
|
||||
{
|
||||
space = m_arm7->space(AS_PROGRAM);
|
||||
}
|
||||
else
|
||||
{
|
||||
space = m_arm9->space(AS_PROGRAM);
|
||||
}
|
||||
|
||||
int cnt = m_dma_cnt[ch];
|
||||
if (cnt == 0)
|
||||
{
|
||||
if (ch == 3)
|
||||
cnt = 0x10000;
|
||||
else
|
||||
cnt = 0x4000;
|
||||
}
|
||||
|
||||
// if (dst >= 0x6000000 && dst <= 0x6017fff)
|
||||
// printf("DMA exec: ch %d from %08x to %08x, mode %04x, count %04x (%s)\n", (int)ch, src, dst, ctrl, cnt, ((ctrl>>10) & 1) ? "32" : "16");
|
||||
|
||||
for (int i = 0; i < cnt; i++)
|
||||
{
|
||||
if ((ctrl>>10) & 1)
|
||||
{
|
||||
src &= 0xfffffffc;
|
||||
dst &= 0xfffffffc;
|
||||
|
||||
// 32-bit
|
||||
space.write_dword(dst, space.read_dword(src));
|
||||
switch (dstadd)
|
||||
{
|
||||
case 0: // increment
|
||||
dst += 4;
|
||||
break;
|
||||
case 1: // decrement
|
||||
dst -= 4;
|
||||
break;
|
||||
case 2: // don't move
|
||||
break;
|
||||
case 3: // increment and reload
|
||||
dst += 4;
|
||||
break;
|
||||
}
|
||||
switch (srcadd)
|
||||
{
|
||||
case 0: // increment
|
||||
src += 4;
|
||||
break;
|
||||
case 1: // decrement
|
||||
src -= 4;
|
||||
break;
|
||||
case 2: // don't move
|
||||
break;
|
||||
case 3: // not used ("Metal Max 2 Kai" expects no increment/decrement)
|
||||
break;
|
||||
}
|
||||
}
|
||||
else
|
||||
{
|
||||
src &= 0xfffffffe;
|
||||
dst &= 0xfffffffe;
|
||||
|
||||
// 16-bit
|
||||
space.write_word(dst, space.read_word(src));
|
||||
switch (dstadd)
|
||||
{
|
||||
case 0: // increment
|
||||
dst += 2;
|
||||
break;
|
||||
case 1: // decrement
|
||||
dst -= 2;
|
||||
break;
|
||||
case 2: // don't move
|
||||
break;
|
||||
case 3: // increment and reload
|
||||
dst += 2;
|
||||
break;
|
||||
}
|
||||
switch (srcadd)
|
||||
{
|
||||
case 0: // increment
|
||||
src += 2;
|
||||
break;
|
||||
case 1: // decrement
|
||||
src -= 2;
|
||||
break;
|
||||
case 2: // don't move
|
||||
break;
|
||||
case 3: // not used (see note in 32-bit version above)
|
||||
break;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
m_dma_src[ch] = src;
|
||||
m_dma_dst[ch] = dst;
|
||||
#endif
|
||||
// printf("settng DMA timer %d for %d cycs (tmr %x)\n", ch, cnt, (uint32_t)m_dma_timer[ch]);
|
||||
// m_dma_timer[ch]->adjust(ATTOTIME_IN_CYCLES(0, cnt), ch);
|
||||
dma_complete(nullptr, ch);
|
||||
}
|
||||
|
||||
TIMER_CALLBACK_MEMBER(nds_state::handle_irq)
|
||||
{
|
||||
request_irq(0, m_if[0]);
|
||||
request_irq(1, m_if[1]);
|
||||
|
||||
m_irq_timer->adjust(attotime::never);
|
||||
}
|
||||
|
||||
void nds_state::request_irq(int cpu, uint32_t int_type)
|
||||
{
|
||||
// set flag for later recovery
|
||||
m_if[cpu] |= int_type;
|
||||
|
||||
printf("request IRQ %08x on CPU %d\n", int_type, cpu);
|
||||
|
||||
// is this specific interrupt enabled?
|
||||
int_type &= m_ie[cpu];
|
||||
if (int_type != 0)
|
||||
{
|
||||
// master enable?
|
||||
if (m_ime[cpu] & 1)
|
||||
{
|
||||
if (cpu == 0)
|
||||
{
|
||||
m_arm9->set_input_line(ARM7_IRQ_LINE, ASSERT_LINE);
|
||||
m_arm9->set_input_line(ARM7_IRQ_LINE, CLEAR_LINE);
|
||||
}
|
||||
else
|
||||
{
|
||||
if (m_arm7halted)
|
||||
{
|
||||
printf("ARM7 unhalting\n");
|
||||
m_arm7->resume(SUSPEND_REASON_HALT);
|
||||
m_arm7halted = false;
|
||||
}
|
||||
|
||||
m_arm7->set_input_line(ARM7_IRQ_LINE, ASSERT_LINE);
|
||||
m_arm7->set_input_line(ARM7_IRQ_LINE, CLEAR_LINE);
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
TIMER_CALLBACK_MEMBER(nds_state::timer_expire)
|
||||
{
|
||||
static const uint32_t tmr_ints[8] = { INT_TM0_OVERFLOW, INT_TM1_OVERFLOW, INT_TM2_OVERFLOW, INT_TM3_OVERFLOW };
|
||||
uintptr_t tmr = (uintptr_t) param;
|
||||
int cpu = (tmr > 4) ? 1 : 0;
|
||||
|
||||
verboselog(*this, 1, "Timer %d expired\n", (int)tmr);
|
||||
|
||||
// "The reload value is copied into the counter only upon following two situations: Automatically upon timer overflows,"
|
||||
// "or when the timer start bit becomes changed from 0 to 1."
|
||||
if (m_timer_recalc[tmr] != 0)
|
||||
{
|
||||
double rate, clocksel, final;
|
||||
attotime time;
|
||||
m_timer_recalc[tmr] = 0;
|
||||
m_timer_regs[tmr] = (m_timer_regs[tmr] & 0xFFFF0000) | (m_timer_reload[tmr] & 0x0000FFFF);
|
||||
rate = 0x10000 - (m_timer_regs[tmr] & 0xffff);
|
||||
clocksel = timer_clks[(m_timer_regs[tmr] >> 16) & 3];
|
||||
final = clocksel / rate;
|
||||
m_timer_hz[tmr] = final;
|
||||
time = attotime::from_hz(final);
|
||||
m_tmr_timer[tmr]->adjust(time, tmr, time);
|
||||
}
|
||||
|
||||
// Handle count-up timing
|
||||
switch (tmr)
|
||||
{
|
||||
case 0:
|
||||
if (m_timer_regs[1] & 0x40000)
|
||||
{
|
||||
m_timer_regs[1] = (( ( m_timer_regs[1] & 0x0000ffff ) + 1 ) & 0x0000ffff) | (m_timer_regs[1] & 0xffff0000);
|
||||
if( ( m_timer_regs[1] & 0x0000ffff ) == 0 )
|
||||
{
|
||||
m_timer_regs[1] |= m_timer_reload[1];
|
||||
if( ( m_timer_regs[1] & 0x400000 ) && ( m_ime[cpu] != 0 ) )
|
||||
{
|
||||
request_irq(cpu, tmr_ints[1]);
|
||||
}
|
||||
if( ( m_timer_regs[2] & 0x40000 ) )
|
||||
{
|
||||
m_timer_regs[2] = (( ( m_timer_regs[2] & 0x0000ffff ) + 1 ) & 0x0000ffff) | (m_timer_regs[2] & 0xffff0000);
|
||||
if( ( m_timer_regs[2] & 0x0000ffff ) == 0 )
|
||||
{
|
||||
m_timer_regs[2] |= m_timer_reload[2];
|
||||
if( ( m_timer_regs[2] & 0x400000 ) && ( m_ime[cpu] != 0 ) )
|
||||
{
|
||||
request_irq(cpu, tmr_ints[2]);
|
||||
}
|
||||
if( ( m_timer_regs[3] & 0x40000 ) )
|
||||
{
|
||||
m_timer_regs[3] = (( ( m_timer_regs[3] & 0x0000ffff ) + 1 ) & 0x0000ffff) | (m_timer_regs[3] & 0xffff0000);
|
||||
if( ( m_timer_regs[3] & 0x0000ffff ) == 0 )
|
||||
{
|
||||
m_timer_regs[3] |= m_timer_reload[3];
|
||||
if( ( m_timer_regs[3] & 0x400000 ) && ( m_ime[cpu] != 0 ) )
|
||||
{
|
||||
request_irq(cpu, tmr_ints[3]);
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
break;
|
||||
case 1:
|
||||
if (m_timer_regs[2] & 0x40000)
|
||||
{
|
||||
m_timer_regs[2] = (( ( m_timer_regs[2] & 0x0000ffff ) + 1 ) & 0x0000ffff) | (m_timer_regs[2] & 0xffff0000);
|
||||
if( ( m_timer_regs[2] & 0x0000ffff ) == 0 )
|
||||
{
|
||||
m_timer_regs[2] |= m_timer_reload[2];
|
||||
if( ( m_timer_regs[2] & 0x400000 ) && ( m_ime[cpu] != 0 ) )
|
||||
{
|
||||
request_irq(cpu, tmr_ints[2]);
|
||||
}
|
||||
if( ( m_timer_regs[3] & 0x40000 ) )
|
||||
{
|
||||
m_timer_regs[3] = (( ( m_timer_regs[3] & 0x0000ffff ) + 1 ) & 0x0000ffff) | (m_timer_regs[3] & 0xffff0000);
|
||||
if( ( m_timer_regs[3] & 0x0000ffff ) == 0 )
|
||||
{
|
||||
m_timer_regs[3] |= m_timer_reload[3];
|
||||
if( ( m_timer_regs[3] & 0x400000 ) && ( m_ime[cpu] != 0 ) )
|
||||
{
|
||||
request_irq(cpu, tmr_ints[3]);
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
break;
|
||||
case 2:
|
||||
if (m_timer_regs[3] & 0x40000)
|
||||
{
|
||||
m_timer_regs[3] = (( ( m_timer_regs[3] & 0x0000ffff ) + 1 ) & 0x0000ffff) | (m_timer_regs[3] & 0xffff0000);
|
||||
if( ( m_timer_regs[3] & 0x0000ffff ) == 0 )
|
||||
{
|
||||
m_timer_regs[3] |= m_timer_reload[3];
|
||||
if( ( m_timer_regs[3] & 0x400000 ) && ( m_ime[cpu] != 0 ) )
|
||||
{
|
||||
request_irq(cpu, tmr_ints[3]);
|
||||
}
|
||||
}
|
||||
}
|
||||
break;
|
||||
}
|
||||
|
||||
// are we supposed to IRQ?
|
||||
if ((m_timer_regs[tmr] & 0x400000) && (m_ime[cpu] != 0))
|
||||
{
|
||||
request_irq(cpu, tmr_ints[tmr & 3]);
|
||||
}
|
||||
}
|
||||
|
||||
static MACHINE_CONFIG_START( nds )
|
||||
MCFG_CPU_ADD("arm7", ARM7, XTAL_33_333MHz)
|
||||
MCFG_CPU_ADD("arm7", ARM7, MASTER_CLOCK)
|
||||
MCFG_CPU_PROGRAM_MAP(nds_arm7_map)
|
||||
|
||||
MCFG_CPU_ADD("arm9", ARM946ES, XTAL_66_6667MHz)
|
||||
MCFG_CPU_ADD("arm9", ARM946ES, MASTER_CLOCK*2)
|
||||
MCFG_ARM_HIGH_VECTORS()
|
||||
MCFG_CPU_PROGRAM_MAP(nds_arm9_map)
|
||||
|
||||
@ -280,5 +1000,5 @@ ROM_START( nds )
|
||||
ROM_LOAD( "firmware.bin", 0x0000, 0x40000, CRC(945f9dc9) SHA1(cfe072921ee3fb93f688743f8beef89043c3e9ad) )
|
||||
ROM_END
|
||||
|
||||
// YEAR NAME PARENT COMPAT MACHINE INPUT STATE INIT COMPANY FULLNAME FLAGS
|
||||
CONS(2004, nds, 0, 0, nds, nds, nds_state, 0, "Nintendo", "DS", MACHINE_IS_SKELETON)
|
||||
// YEAR NAME PARENT COMPAT MACHINE INPUT STATE INIT COMPANY FULLNAME FLAGS
|
||||
CONS(2004, nds, 0, 0, nds, nds, nds_state, 0, "Nintendo", "DS", MACHINE_NOT_WORKING | MACHINE_NO_SOUND)
|
||||
|
@ -1,5 +1,5 @@
|
||||
// license:BSD-3-Clause
|
||||
// copyright-holders:Ryan Holtz
|
||||
// copyright-holders:Ryan Holtz, R. Belmont
|
||||
#pragma once
|
||||
#ifndef INCLUDES_NDS_H
|
||||
#define INCLUDES_NDS_H
|
||||
@ -48,25 +48,61 @@ protected:
|
||||
required_shared_ptr<uint32_t> m_arm7ram;
|
||||
|
||||
enum {
|
||||
IPCSYNC_OFFSET = 0x180/4,
|
||||
GAMECARD_BUS_CTRL_OFFSET = 0x1a4/4,
|
||||
WRAMSTAT_OFFSET = 0x241/4,
|
||||
VRAMCNT_A_OFFSET = 0x240/4,
|
||||
WRAMCNT_OFFSET = 0x244/4,
|
||||
VRAMCNT_H_OFFSET = 0x248/4,
|
||||
POSTFLG_OFFSET = 0x300/4,
|
||||
TIMER_OFFSET = (0x100/4),
|
||||
RTC_OFFSET = (0x138/4),
|
||||
IPCSYNC_OFFSET = (0x180/4),
|
||||
AUX_SPI_CNT_OFFSET = (0x1a0/4),
|
||||
GAMECARD_BUS_CTRL_OFFSET = (0x1a4/4),
|
||||
GAMECARD_DATA_OFFSET = (0x1a8/4),
|
||||
GAMECARD_DATA_2_OFFSET = (0x1ac/4),
|
||||
SPI_CTRL_OFFSET = (0x1c0/4),
|
||||
IME_OFFSET = (0x208/4),
|
||||
IE_OFFSET = (0x210/4),
|
||||
IF_OFFSET = (0x214/4),
|
||||
WRAMSTAT_OFFSET = (0x241/4),
|
||||
VRAMCNT_A_OFFSET = (0x240/4),
|
||||
WRAMCNT_OFFSET = (0x244/4),
|
||||
VRAMCNT_H_OFFSET = (0x248/4),
|
||||
POSTFLG_OFFSET = (0x300/4),
|
||||
GAMECARD_DATA_IN_OFFSET = (0x100010/4),
|
||||
POSTFLG_PBF_SHIFT = 0,
|
||||
POSTFLG_RAM_SHIFT = 1,
|
||||
POSTFLG_PBF_MASK = (1 << POSTFLG_PBF_SHIFT),
|
||||
POSTFLG_RAM_MASK = (1 << POSTFLG_RAM_SHIFT),
|
||||
GAMECARD_DATA_READY = (1 << 23),
|
||||
GAMECARD_BLOCK_BUSY = (1 << 31)
|
||||
};
|
||||
|
||||
uint32_t m_arm7_postflg;
|
||||
uint32_t m_arm9_postflg;
|
||||
uint16_t m_arm7_ipcsync, m_arm9_ipcsync;
|
||||
uint32_t m_gamecard_ctrl, m_cartdata_len;
|
||||
uint32_t m_ime[2], m_ie[2], m_if[2];
|
||||
uint16_t m_arm7_ipcsync, m_arm9_ipcsync, m_spicnt;
|
||||
uint8_t m_WRAM[0x8000];
|
||||
uint8_t m_wramcnt;
|
||||
uint8_t m_vramcnta, m_vramcntb, m_vramcntc, m_vramcntd, m_vramcnte, m_vramcntf, m_vramcntg, m_vramcnth, m_vramcnti;
|
||||
bool m_arm7halted;
|
||||
|
||||
// DMA
|
||||
emu_timer *m_dma_timer[8];
|
||||
uint32_t m_dma_src[8];
|
||||
uint32_t m_dma_dst[8];
|
||||
uint16_t m_dma_cnt[8];
|
||||
|
||||
// Timers
|
||||
uint32_t m_timer_regs[8];
|
||||
uint16_t m_timer_reload[8];
|
||||
int m_timer_recalc[8];
|
||||
double m_timer_hz[8];
|
||||
|
||||
emu_timer *m_tmr_timer[8], *m_irq_timer;
|
||||
|
||||
TIMER_CALLBACK_MEMBER(dma_complete);
|
||||
TIMER_CALLBACK_MEMBER(timer_expire);
|
||||
TIMER_CALLBACK_MEMBER(handle_irq);
|
||||
|
||||
void request_irq(int which_cpu, uint32_t int_type);
|
||||
void dma_exec(int ch);
|
||||
};
|
||||
|
||||
#endif // INCLUDES_NDS_H
|
||||
|
Loading…
Reference in New Issue
Block a user