#include "shared.h"
static unsigned int s68k_read_bus_8(unsigned int address)
{
#ifdef LOGERROR
error("[SUB 68k] Unused read8 %08X (%08X)\n", address, s68k.pc);
#endif
address = s68k.pc | (address & 1);
return READ_BYTE(s68k.memory_map[((address)>>16)&0xff].base, (address) & 0xffff);
}
static unsigned int s68k_read_bus_16(unsigned int address)
{
#ifdef LOGERROR
error("[SUB 68k] Unused read16 %08X (%08X)\n", address, s68k.pc);
#endif
address = s68k.pc;
return *(uint16 *)(s68k.memory_map[((address)>>16)&0xff].base + ((address) & 0xffff));
}
static void s68k_unused_8_w(unsigned int address, unsigned int data)
{
#ifdef LOGERROR
error("[SUB 68k] Unused write8 %08X = %02X (%08X)\n", address, data, s68k.pc);
#endif
}
static void s68k_unused_16_w(unsigned int address, unsigned int data)
{
#ifdef LOGERROR
error("[SUB 68k] Unused write16 %08X = %04X (%08X)\n", address, data, s68k.pc);
#endif
}
void prg_ram_dma_w(unsigned int words)
{
uint16 data;
uint16 src_index = cdc.dac.w & 0x3ffe;
uint32 dst_index = (scd.regs[0x0a>>1].w << 3) & 0x7fffe;
scd.regs[0x0a>>1].w += (words >> 2);
cdc.dac.w += (words << 1);
if (dst_index < (scd.regs[0x02>>1].byte.h << 9))
{
return;
}
while (words--)
{
data = *(uint16 *)(cdc.ram + src_index);
#ifdef LSB_FIRST
data = ((data >> 8) | (data << 8)) & 0xffff;
#endif
*(uint16 *)(scd.prg_ram + dst_index) = data ;
src_index = (src_index + 2) & 0x3ffe;
dst_index = (dst_index + 2) & 0x7fffe;
}
}
static void prg_ram_write_byte(unsigned int address, unsigned int data)
{
address &= 0x7ffff;
if (address >= (scd.regs[0x02>>1].byte.h << 9))
{
WRITE_BYTE(scd.prg_ram, address, data);
return;
}
#ifdef LOGERROR
error("[SUB 68k] PRG-RAM protected write8 %08X = %02X (%08X)\n", address, data, s68k.pc);
#endif
}
static void prg_ram_write_word(unsigned int address, unsigned int data)
{
address &= 0x7fffe;
if (address >= (scd.regs[0x02>>1].byte.h << 9))
{
*(uint16 *)(scd.prg_ram + address) = data;
return;
}
#ifdef LOGERROR
error("[SUB 68k] PRG-RAM protected write16 %08X = %02X (%08X)\n", address, data, s68k.pc);
#endif
}
static unsigned int bram_read_byte(unsigned int address)
{
if (address & 1)
{
return scd.bram[(address >> 1) & 0x1fff];
}
return 0xff;
}
static unsigned int bram_read_word(unsigned int address)
{
return (scd.bram[(address >> 1) & 0x1fff] | 0xff00);
}
static void bram_write_byte(unsigned int address, unsigned int data)
{
if (address & 1)
{
scd.bram[(address >> 1) & 0x1fff] = data;
}
}
static void bram_write_word(unsigned int address, unsigned int data)
{
scd.bram[(address >> 1) & 0x1fff] = data & 0xff;
}
static void s68k_poll_detect(unsigned int reg_mask)
{
if (s68k.poll.detected & reg_mask)
{
if (s68k.cycles <= s68k.poll.cycle)
{
if (s68k.pc == s68k.poll.pc)
{
if (s68k.poll.detected & 1)
{
s68k.cycles = s68k.cycle_end;
s68k.stopped = reg_mask;
#ifdef LOG_SCD
error("s68k stopped from %d cycles\n", s68k.cycles);
#endif
}
else
{
s68k.poll.detected |= 1;
s68k.poll.cycle = s68k.cycles + 392;
}
}
return;
}
}
else
{
s68k.poll.detected = reg_mask;
}
s68k.poll.cycle = s68k.cycles + 392;
s68k.poll.pc = s68k.pc;
}
static void s68k_poll_sync(unsigned int reg_mask)
{
unsigned int cycles = (s68k.cycles * MCYCLES_PER_LINE) / SCYCLES_PER_LINE;
if (!m68k.stopped)
{
m68k_run(cycles);
}
if (m68k.stopped & reg_mask)
{
m68k.cycles = cycles;
m68k.stopped = 0;
#ifdef LOG_SCD
error("m68k started from %d cycles\n", cycles);
#endif
}
s68k.poll.detected &= ~reg_mask;
m68k.poll.detected &= ~reg_mask;
}
static unsigned int scd_read_byte(unsigned int address)
{
if (address < 0xff8000)
{
if (address & 1)
{
return pcm_read((address >> 1) & 0x1fff);
}
return s68k_read_bus_8(address);
}
#ifdef LOG_SCD
error("[%d][%d]read byte CD register %X (%X)\n", v_counter, s68k.cycles, address, s68k.pc);
#endif
if (address == 0xff8003)
{
s68k_poll_detect(1<<0x03);
return scd.regs[0x03>>1].byte.l;
}
if (address == 0xff800e)
{
s68k_poll_detect(1<<0x0e);
return scd.regs[0x0e>>1].byte.h;
}
if (address == 0xff8004)
{
s68k_poll_detect(1<<0x04);
return scd.regs[0x04>>1].byte.h;
}
if (address == 0xff8058)
{
s68k_poll_detect(1<<0x08);
return scd.regs[0x58>>1].byte.h;
}
if (address == 0xff8007)
{
unsigned int data = cdc_reg_r();
#ifdef LOG_CDC
error("CDC register %X read 0x%02X (%X)\n", scd.regs[0x04>>1].byte.l & 0x0F, data, s68k.pc);
#endif
return data;
}
if (address == 0xff8000)
{
return scd.regs[0x06>>1].byte.h;
}
if (address == 0xff8001)
{
return 0x01;
}
if ((address >= 0xff8050) && (address <= 0xff8056))
{
uint8 bits = (scd.regs[0x4e>>1].w >> (((address & 6) ^ 6) << 1)) << 2;
uint8 code = scd.regs[0x4c>>1].byte.l;
uint16 data = (code >> (bits & 4)) & 0x0f;
bits = bits >> 1;
data = data | (((code >> (bits & 4)) << 4) & 0xf0);
bits = bits >> 1;
data = data | (((code >> (bits & 4)) << 8) & 0xf00);
bits = bits >> 1;
data = data | (((code >> (bits & 4)) << 12) & 0xf000);
return (address & 1) ? (data & 0xff) : (data >> 8);
}
if ((address & 0x1f0) == 0x10)
{
s68k_poll_detect(1 << (address & 0x1f));
}
if (address & 1)
{
return scd.regs[(address >> 1) & 0xff].byte.l;
}
return scd.regs[(address >> 1) & 0xff].byte.h;
}
static unsigned int scd_read_word(unsigned int address)
{
if (address < 0xff8000)
{
return pcm_read((address >> 1) & 0x1fff);
}
#ifdef LOG_SCD
error("[%d][%d]read word CD register %X (%X)\n", v_counter, s68k.cycles, address, s68k.pc);
#endif
if (address == 0xff8002)
{
s68k_poll_detect(1<<0x03);
return scd.regs[0x03>>1].w;
}
if (address == 0xff8008)
{
return cdc_host_r();
}
if (address == 0xff8000)
{
return scd.regs[0x06>>1].w;
}
if (address == 0xff800c)
{
return (scd.regs[0x0c>>1].w + ((s68k.cycles - scd.stopwatch) / TIMERS_SCYCLES_RATIO)) & 0xfff;
}
if ((address >= 0xff8050) && (address <= 0xff8056))
{
uint8 bits = (scd.regs[0x4e>>1].w >> (((address & 6) ^ 6) << 1)) << 2;
uint8 code = scd.regs[0x4c>>1].byte.l;
uint16 data = (code >> (bits & 4)) & 0x0f;
bits = bits >> 1;
data = data | (((code >> (bits & 4)) << 4) & 0xf0);
bits = bits >> 1;
data = data | (((code >> (bits & 4)) << 8) & 0xf00);
bits = bits >> 1;
data = data | (((code >> (bits & 4)) << 12) & 0xf000);
return data;
}
if ((address & 0x1f0) == 0x10)
{
if (!m68k.stopped)
{
unsigned int cycles = (s68k.cycles * MCYCLES_PER_LINE) / SCYCLES_PER_LINE;
m68k_run(cycles);
}
s68k_poll_detect(3 << (address & 0x1e));
}
return scd.regs[(address >> 1) & 0xff].w;
}
INLINE void word_ram_switch(uint8 mode)
{
int i;
uint16 *ptr1 = (uint16 *)(scd.word_ram_2M);
uint16 *ptr2 = (uint16 *)(scd.word_ram[0]);
uint16 *ptr3 = (uint16 *)(scd.word_ram[1]);
if (mode & 0x04)
{
for (i=0; i<0x10000; i++)
{
*ptr2++=*ptr1++;
*ptr3++=*ptr1++;
}
}
else
{
for (i=0; i<0x10000; i++)
{
*ptr1++=*ptr2++;
*ptr1++=*ptr3++;
}
for (i=scd.cartridge.boot+0x20; i<scd.cartridge.boot+0x24; i++)
{
m68k.memory_map[i].base = scd.word_ram_2M + ((i & 0x03) << 16);
m68k.memory_map[i].read8 = NULL;
m68k.memory_map[i].read16 = NULL;
m68k.memory_map[i].write8 = NULL;
m68k.memory_map[i].write16 = NULL;
zbank_memory_map[i].read = NULL;
zbank_memory_map[i].write = NULL;
}
for (i=0x08; i<0x0c; i++)
{
s68k.memory_map[i].read8 = NULL;
s68k.memory_map[i].read16 = NULL;
s68k.memory_map[i].write8 = NULL;
s68k.memory_map[i].write16 = NULL;
}
for (i=0x0c; i<0x0e; i++)
{
s68k.memory_map[i].read8 = s68k_read_bus_8;
s68k.memory_map[i].read16 = s68k_read_bus_16;
s68k.memory_map[i].write8 = s68k_unused_8_w;
s68k.memory_map[i].write16 = s68k_unused_16_w;
}
}
}
static void scd_write_byte(unsigned int address, unsigned int data)
{
if (address < 0xff8000)
{
if (address & 1)
{
pcm_write((address >> 1) & 0x1fff, data);
return;
}
s68k_unused_8_w(address, data);
return;
}
#ifdef LOG_SCD
error("[%d][%d]write byte CD register %X -> 0x%02x (%X)\n", v_counter, s68k.cycles, address, data, s68k.pc);
#endif
switch (address & 0x1ff)
{
case 0x00:
{
scd.regs[0x06 >> 1].byte.h = data;
return;
}
case 0x01:
{
if (!(data & 0x01))
{
scd_reset(0);
}
return;
}
case 0x03:
{
s68k_poll_sync(1<<0x03);
if ((data ^ scd.regs[0x03 >> 1].byte.l) & 0x05)
{
int i;
if (data & 0x04)
{
if (!(scd.regs[0x03 >> 1].byte.l & 0x04))
{
word_ram_switch(0x04);
}
if (data & 0x01)
{
for (i=scd.cartridge.boot+0x20; i<scd.cartridge.boot+0x22; i++)
{
m68k.memory_map[i].base = scd.word_ram[1] + ((i & 0x01) << 16);
}
for (i=scd.cartridge.boot+0x22; i<scd.cartridge.boot+0x24; i++)
{
m68k.memory_map[i].read8 = cell_ram_1_read8;
m68k.memory_map[i].read16 = cell_ram_1_read16;
m68k.memory_map[i].write8 = cell_ram_1_write8;
m68k.memory_map[i].write16 = cell_ram_1_write16;
zbank_memory_map[i].read = cell_ram_1_read8;
zbank_memory_map[i].write = cell_ram_1_write8;
}
for (i=0x08; i<0x0c; i++)
{
s68k.memory_map[i].read8 = dot_ram_0_read8;
s68k.memory_map[i].read16 = dot_ram_0_read16;
s68k.memory_map[i].write8 = dot_ram_0_write8;
s68k.memory_map[i].write16 = dot_ram_0_write16;
}
for (i=0x0c; i<0x0e; i++)
{
s68k.memory_map[i].base = scd.word_ram[0] + ((i & 0x01) << 16);
s68k.memory_map[i].read8 = NULL;
s68k.memory_map[i].read16 = NULL;
s68k.memory_map[i].write8 = NULL;
s68k.memory_map[i].write16 = NULL;
}
scd.dmna = 0;
}
else
{
for (i=scd.cartridge.boot+0x20; i<scd.cartridge.boot+0x22; i++)
{
m68k.memory_map[i].base = scd.word_ram[0] + ((i & 0x01) << 16);
}
for (i=scd.cartridge.boot+0x22; i<scd.cartridge.boot+0x24; i++)
{
m68k.memory_map[i].read8 = cell_ram_0_read8;
m68k.memory_map[i].read16 = cell_ram_0_read16;
m68k.memory_map[i].write8 = cell_ram_0_write8;
m68k.memory_map[i].write16 = cell_ram_0_write16;
zbank_memory_map[i].read = cell_ram_0_read8;
zbank_memory_map[i].write = cell_ram_0_write8;
}
for (i=0x08; i<0x0c; i++)
{
s68k.memory_map[i].read8 = dot_ram_1_read8;
s68k.memory_map[i].read16 = dot_ram_1_read16;
s68k.memory_map[i].write8 = dot_ram_1_write8;
s68k.memory_map[i].write16 = dot_ram_1_write16;
}
for (i=0x0c; i<0x0e; i++)
{
s68k.memory_map[i].base = scd.word_ram[1] + ((i & 0x01) << 16);
s68k.memory_map[i].read8 = NULL;
s68k.memory_map[i].read16 = NULL;
s68k.memory_map[i].write8 = NULL;
s68k.memory_map[i].write16 = NULL;
}
}
scd.regs[0x02 >> 1].byte.l = (scd.regs[0x02 >> 1].byte.l & ~0x1f) | (data & 0x1d);
return;
}
else
{
if (scd.regs[0x02 >> 1].byte.l & 0x04)
{
word_ram_switch(0x00);
data |= ~scd.dmna & 0x01;
if (!(data & 0x01))
{
data |= 0x02;
scd.regs[0x02 >> 1].byte.l = (scd.regs[0x02 >> 1].byte.l & ~0x1f) | (data & 0x1f);
return;
}
}
if (data & 0x01)
{
scd.dmna = 0;
scd.regs[0x02 >> 1].byte.l = (scd.regs[0x02 >> 1].byte.l & ~0x1f) | (data & 0x1d);
return;
}
}
}
scd.regs[0x02 >> 1].byte.l = (scd.regs[0x02 >> 1].byte.l & ~0x1c) | (data & 0x1c);
return;
}
case 0x07:
{
cdc_reg_w(data);
return;
}
case 0x0e:
case 0x0f:
{
s68k_poll_sync(1<<0x0f);
scd.regs[0x0f>>1].byte.l = data;
return;
}
case 0x31:
{
scd.timer = data * TIMERS_SCYCLES_RATIO;
if (data)
{
scd.timer += (s68k.cycles - scd.cycles);
}
scd.regs[0x30>>1].byte.l = data;
return;
}
case 0x33:
{
scd.regs[0x32>>1].byte.l = data;
scd.regs[0x00].byte.h = (scd.regs[0x00].byte.h & 0x7f) | ((data & 0x04) << 5);
scd.pending &= ~(data & 0x02);
s68k_update_irq((scd.pending & data) >> 1);
return;
}
case 0x37:
{
if ((data & 0x04) && !(scd.regs[0x37>>1].byte.l & 0x04))
{
cdd.cycles = (scd.cycles - s68k.cycles) * 3;
scd.pending |= (1 << 4);
if (scd.regs[0x32>>1].byte.l & 0x10)
{
s68k_update_irq((scd.pending & scd.regs[0x32>>1].byte.l) >> 1);
}
}
scd.regs[0x37>>1].byte.l = data;
return;
}
default:
{
if ((address & 0xf0) == 0x20)
{
s68k_poll_sync(1 << ((address - 0x10) & 0x1f));
}
if (address & 1)
{
scd.regs[(address >> 1) & 0xff].byte.l = data;
return;
}
scd.regs[(address >> 1) & 0xff].byte.h = data;
return;
}
}
}
static void scd_write_word(unsigned int address, unsigned int data)
{
if (address < 0xff8000)
{
pcm_write((address >> 1) & 0x1fff, data);
return;
}
#ifdef LOG_SCD
error("[%d][%d]write word CD register %X -> 0x%04x (%X)\n", v_counter, s68k.cycles, address, data, s68k.pc);
#endif
switch (address & 0x1fe)
{
case 0x00:
{
scd.regs[0x06>>1].byte.h = data >> 8;
if (!(data & 0x01))
{
scd_reset(0);
}
return;
}
case 0x02:
{
s68k_poll_sync(1<<0x03);
if ((data ^ scd.regs[0x03>>1].byte.l) & 0x05)
{
int i;
if (data & 0x04)
{
if (!(scd.regs[0x03 >> 1].byte.l & 0x04))
{
word_ram_switch(0x04);
}
if (data & 0x01)
{
for (i=scd.cartridge.boot+0x20; i<scd.cartridge.boot+0x22; i++)
{
m68k.memory_map[i].base = scd.word_ram[1] + ((i & 0x01) << 16);
}
for (i=scd.cartridge.boot+0x22; i<scd.cartridge.boot+0x24; i++)
{
m68k.memory_map[i].read8 = cell_ram_1_read8;
m68k.memory_map[i].read16 = cell_ram_1_read16;
m68k.memory_map[i].write8 = cell_ram_1_write8;
m68k.memory_map[i].write16 = cell_ram_1_write16;
zbank_memory_map[i].read = cell_ram_1_read8;
zbank_memory_map[i].write = cell_ram_1_write8;
}
for (i=0x08; i<0x0c; i++)
{
s68k.memory_map[i].read8 = dot_ram_0_read8;
s68k.memory_map[i].read16 = dot_ram_0_read16;
s68k.memory_map[i].write8 = dot_ram_0_write8;
s68k.memory_map[i].write16 = dot_ram_0_write16;
}
for (i=0x0c; i<0x0e; i++)
{
s68k.memory_map[i].base = scd.word_ram[0] + ((i & 0x01) << 16);
s68k.memory_map[i].read8 = NULL;
s68k.memory_map[i].read16 = NULL;
s68k.memory_map[i].write8 = NULL;
s68k.memory_map[i].write16 = NULL;
}
scd.dmna = 0;
}
else
{
for (i=scd.cartridge.boot+0x20; i<scd.cartridge.boot+0x22; i++)
{
m68k.memory_map[i].base = scd.word_ram[0] + ((i & 0x01) << 16);
}
for (i=scd.cartridge.boot+0x22; i<scd.cartridge.boot+0x24; i++)
{
m68k.memory_map[i].read8 = cell_ram_0_read8;
m68k.memory_map[i].read16 = cell_ram_0_read16;
m68k.memory_map[i].write8 = cell_ram_0_write8;
m68k.memory_map[i].write16 = cell_ram_0_write16;
zbank_memory_map[i].read = cell_ram_0_read8;
zbank_memory_map[i].write = cell_ram_0_write8;
}
for (i=0x08; i<0x0c; i++)
{
s68k.memory_map[i].read8 = dot_ram_1_read8;
s68k.memory_map[i].read16 = dot_ram_1_read16;
s68k.memory_map[i].write8 = dot_ram_1_write8;
s68k.memory_map[i].write16 = dot_ram_1_write16;
}
for (i=0x0c; i<0x0e; i++)
{
s68k.memory_map[i].base = scd.word_ram[1] + ((i & 0x01) << 16);
s68k.memory_map[i].read8 = NULL;
s68k.memory_map[i].read16 = NULL;
s68k.memory_map[i].write8 = NULL;
s68k.memory_map[i].write16 = NULL;
}
}
scd.regs[0x03>>1].byte.l = (scd.regs[0x03>>1].byte.l & ~0x1f) | (data & 0x1d);
return;
}
else
{
if (scd.regs[0x03>>1].byte.l & 0x04)
{
word_ram_switch(0x00);
data |= ~scd.dmna & 0x01;
if (!(data & 0x01))
{
data |= 0x02;
scd.regs[0x03>>1].byte.l = (scd.regs[0x03>>1].byte.l & ~0x1f) | (data & 0x1f);
return;
}
}
if (data & 0x01)
{
scd.dmna = 0;
scd.regs[0x03>>1].byte.l = (scd.regs[0x03>>1].byte.l & ~0x1f) | (data & 0x1d);
return;
}
}
}
scd.regs[0x03>>1].byte.l = (scd.regs[0x03>>1].byte.l & ~0x1c) | (data & 0x1c);
return;
}
case 0x06:
{
cdc_reg_w(data);
return;
}
case 0x0c:
{
int ticks = (s68k.cycles - scd.stopwatch) / TIMERS_SCYCLES_RATIO;
scd.stopwatch += (ticks * TIMERS_SCYCLES_RATIO);
scd.regs[0x0c>>1].w = 0;
return;
}
case 0x0e:
{
s68k_poll_sync(1<<0x0f);
scd.regs[0x0f>>1].byte.l = data & 0xff;
return;
}
case 0x30:
{
data &= 0xff;
scd.timer = data * TIMERS_SCYCLES_RATIO;
if (data)
{
scd.timer += (s68k.cycles - scd.cycles);
}
scd.regs[0x30>>1].byte.l = data;
return;
}
case 0x32:
{
data &= 0xff;
scd.regs[0x32>>1].byte.l = data;
scd.regs[0x00].byte.h = (scd.regs[0x00].byte.h & 0x7f) | ((data & 0x04) << 5);
scd.pending &= ~(data & 0x02);
s68k_update_irq((scd.pending & data) >> 1);
return;
}
case 0x4a:
{
scd.regs[0x4a>>1].w = 0;
cdd_process();
#ifdef LOG_CDD
error("CDD command: %02x %02x %02x %02x %02x %02x %02x %02x\n",scd.regs[0x42>>1].byte.h, scd.regs[0x42>>1].byte.l, scd.regs[0x44>>1].byte.h, scd.regs[0x44>>1].byte.l, scd.regs[0x46>>1].byte.h, scd.regs[0x46>>1].byte.l, scd.regs[0x48>>1].byte.h, scd.regs[0x48>>1].byte.l);
error("CDD status: %02x %02x %02x %02x %02x %02x %02x %02x %02x %02x\n",scd.regs[0x38>>1].byte.h, scd.regs[0x38>>1].byte.l, scd.regs[0x3a>>1].byte.h, scd.regs[0x3a>>1].byte.l, scd.regs[0x3c>>1].byte.h, scd.regs[0x3c>>1].byte.l, scd.regs[0x3e>>1].byte.h, scd.regs[0x3e>>1].byte.l, scd.regs[0x40>>1].byte.h, scd.regs[0x40>>1].byte.l);
#endif
break;
}
case 0x66:
{
scd.regs[0x66>>1].w = data;
gfx_start(data, s68k.cycles);
return;
}
default:
{
if ((address & 0xf0) == 0x20)
{
s68k_poll_sync(3 << ((address - 0x10) & 0x1e));
}
scd.regs[(address >> 1) & 0xff].w = data;
return;
}
}
}
void scd_init(void)
{
int i;
uint8 base = scd.cartridge.boot;
cd_cart_init();
for (i=base; i<base+0x20; i++)
{
if (i & 2)
{
m68k.memory_map[i].base = scd.prg_ram + ((i & 1) << 16);
m68k.memory_map[i].read8 = NULL;
m68k.memory_map[i].read16 = NULL;
m68k.memory_map[i].write8 = NULL;
m68k.memory_map[i].write16 = NULL;
zbank_memory_map[i].read = NULL;
zbank_memory_map[i].write = NULL;
}
else
{
m68k.memory_map[i].base = scd.bootrom + ((i & 1) << 16);
m68k.memory_map[i].read8 = NULL;
m68k.memory_map[i].read16 = NULL;
m68k.memory_map[i].write8 = m68k_unused_8_w;
m68k.memory_map[i].write16 = m68k_unused_16_w;
zbank_memory_map[i].read = NULL;
zbank_memory_map[i].write = zbank_unused_w;
}
}
for (i=base+0x20; i<base+0x40; i++)
{
m68k.memory_map[i].base = scd.word_ram_2M + ((i & 3) << 16);
m68k.memory_map[i].read8 = NULL;
m68k.memory_map[i].read16 = NULL;
m68k.memory_map[i].write8 = NULL;
m68k.memory_map[i].write16 = NULL;
zbank_memory_map[i].read = NULL;
zbank_memory_map[i].write = NULL;
}
for (i=0x00; i<0x08; i++)
{
s68k.memory_map[i].base = scd.prg_ram + (i << 16);
s68k.memory_map[i].read8 = NULL;
s68k.memory_map[i].read16 = NULL;
s68k.memory_map[i].write8 = (i < 0x02) ? prg_ram_write_byte : NULL;
s68k.memory_map[i].write16 = (i < 0x02) ? prg_ram_write_word : NULL;
}
for (i=0x08; i<0x0c; i++)
{
s68k.memory_map[i].base = scd.word_ram_2M + ((i & 3) << 16);
s68k.memory_map[i].read8 = NULL;
s68k.memory_map[i].read16 = NULL;
s68k.memory_map[i].write8 = NULL;
s68k.memory_map[i].write16 = NULL;
}
for (i=0x0c; i<0xfd; i++)
{
s68k.memory_map[i].base = scd.word_ram_2M + ((i & 3) << 16);
s68k.memory_map[i].read8 = s68k_read_bus_8;
s68k.memory_map[i].read16 = s68k_read_bus_16;
s68k.memory_map[i].write8 = s68k_unused_8_w;
s68k.memory_map[i].write16 = s68k_unused_16_w;
}
for (i=0xfd; i<0xff; i++)
{
s68k.memory_map[i].base = NULL;
s68k.memory_map[i].read8 = bram_read_byte;
s68k.memory_map[i].read16 = bram_read_word;
s68k.memory_map[i].write8 = bram_write_byte;
s68k.memory_map[i].write16 = bram_write_word;
}
s68k.memory_map[0xff].base = NULL;
s68k.memory_map[0xff].read8 = scd_read_byte;
s68k.memory_map[0xff].read16 = scd_read_word;
s68k.memory_map[0xff].write8 = scd_write_byte;
s68k.memory_map[0xff].write16 = scd_write_word;
cdc_init();
gfx_init();
memset(scd.prg_ram, 0x00, sizeof(scd.prg_ram));
memset(scd.word_ram, 0x00, sizeof(scd.word_ram));
memset(scd.word_ram_2M, 0x00, sizeof(scd.word_ram_2M));
memset(scd.bram, 0x00, sizeof(scd.bram));
}
void scd_reset(int hard)
{
if (hard)
{
int i;
memset(scd.regs, 0, sizeof(scd.regs));
scd.dmna = 0;
*(uint16 *)(m68k.memory_map[scd.cartridge.boot].base + 0x70) = 0x00FF;
*(uint16 *)(m68k.memory_map[scd.cartridge.boot].base + 0x72) = 0xFFFF;
scd.regs[0x00>>1].w = 0x0002;
scd.regs[0x02>>1].w = 0x0001;
word_ram_switch(0);
for (i=scd.cartridge.boot+0x02; i<scd.cartridge.boot+0x20; i+=4)
{
m68k.memory_map[i].base = scd.prg_ram;
m68k.memory_map[i+1].base = scd.prg_ram + 0x10000;
}
s68k.cycles = 0;
s68k_pulse_reset();
s68k_pulse_halt();
}
else
{
memset(&scd.regs[0x04>>1], 0, sizeof(scd.regs) - 4);
}
scd.regs[0x08>>1].w = 0xffff;
scd.regs[0x0a>>1].w = 0xffff;
scd.regs[0x36>>1].w = 0x0100;
scd.regs[0x40>>1].w = 0x000f;
scd.regs[0x42>>1].w = 0xffff;
scd.regs[0x44>>1].w = 0xffff;
scd.regs[0x46>>1].w = 0xffff;
scd.regs[0x48>>1].w = 0xffff;
scd.regs[0x4a>>1].w = 0xffff;
scd.regs[0x06>>1].byte.l = 0x01;
scd.timer = 0;
scd.stopwatch = 0;
scd.cycles = 0;
scd.pending = 0;
memset(&m68k.poll, 0, sizeof(m68k.poll));
memset(&s68k.poll, 0, sizeof(s68k.poll));
cdd_reset();
cdc_reset();
gfx_reset();
pcm_reset();
}
void scd_update(unsigned int cycles)
{
if (cdc.dma_w)
{
cdc_dma_update();
}
do
{
m68k_run(cycles);
s68k_run(scd.cycles + SCYCLES_PER_LINE);
}
while ((m68k.cycles < cycles) || (s68k.cycles < (scd.cycles + SCYCLES_PER_LINE)));
scd.cycles += SCYCLES_PER_LINE;
cdd.cycles += (SCYCLES_PER_LINE * 3);
if (cdd.cycles >= (500000 * 4))
{
cdd.cycles -= (500000 * 4);
cdd_update();
if (!(scd.regs[0x4a>>1].byte.l & 0xf0))
{
scd.regs[0x4a>>1].byte.l = 0xf0;
scd.pending |= (1 << 4);
if (scd.regs[0x32>>1].byte.l & 0x10)
{
s68k_update_irq((scd.pending & scd.regs[0x32>>1].byte.l) >> 1);
}
}
}
if (scd.timer)
{
scd.timer -= SCYCLES_PER_LINE;
if (scd.timer <= 0)
{
scd.timer += (scd.regs[0x30>>1].byte.l * TIMERS_SCYCLES_RATIO);
if (scd.regs[0x32>>1].byte.l & 0x08)
{
scd.pending |= (1 << 3);
s68k_update_irq((scd.pending & scd.regs[0x32>>1].byte.l) >> 1);
}
}
}
if (scd.regs[0x58>>1].byte.h & 0x80)
{
gfx_update(scd.cycles);
}
}
void scd_end_frame(unsigned int cycles)
{
int ticks = (cycles - scd.stopwatch) / TIMERS_SCYCLES_RATIO;
scd.regs[0x0c>>1].w = (scd.regs[0x0c>>1].w + ticks) & 0xfff;
scd.stopwatch += (ticks * TIMERS_SCYCLES_RATIO) - cycles;
s68k.cycles -= cycles;
gfx.cycles -= cycles;
m68k.poll.cycle = 0;
s68k.poll.cycle = 0;
}
int scd_context_save(uint8 *state)
{
uint16 tmp16;
uint32 tmp32;
int bufferptr = 0;
save_param(scd.regs, sizeof(scd.regs));
save_param(&scd.cycles, sizeof(scd.cycles));
save_param(&scd.timer, sizeof(scd.timer));
save_param(&scd.pending, sizeof(scd.pending));
save_param(&scd.dmna, sizeof(scd.dmna));
bufferptr += gfx_context_save(&state[bufferptr]);
bufferptr += cdc_context_save(&state[bufferptr]);
bufferptr += cdd_context_save(&state[bufferptr]);
bufferptr += pcm_context_save(&state[bufferptr]);
save_param(scd.prg_ram, sizeof(scd.prg_ram));
if (scd.regs[0x03>>1].byte.l & 0x04)
{
save_param(scd.word_ram, sizeof(scd.word_ram));
}
else
{
save_param(scd.word_ram_2M, sizeof(scd.word_ram_2M));
}
save_param(&m68k.poll, sizeof(m68k.poll));
save_param(&s68k.poll, sizeof(s68k.poll));
tmp16 = *(uint16 *)(m68k.memory_map[scd.cartridge.boot].base + 0x72);
save_param(&tmp16, 2);
save_param(&s68k.cycles, sizeof(s68k.cycles));
save_param(&s68k.int_level, sizeof(s68k.int_level));
save_param(&s68k.stopped, sizeof(s68k.stopped));
tmp32 = s68k_get_reg(M68K_REG_D0); save_param(&tmp32, 4);
tmp32 = s68k_get_reg(M68K_REG_D1); save_param(&tmp32, 4);
tmp32 = s68k_get_reg(M68K_REG_D2); save_param(&tmp32, 4);
tmp32 = s68k_get_reg(M68K_REG_D3); save_param(&tmp32, 4);
tmp32 = s68k_get_reg(M68K_REG_D4); save_param(&tmp32, 4);
tmp32 = s68k_get_reg(M68K_REG_D5); save_param(&tmp32, 4);
tmp32 = s68k_get_reg(M68K_REG_D6); save_param(&tmp32, 4);
tmp32 = s68k_get_reg(M68K_REG_D7); save_param(&tmp32, 4);
tmp32 = s68k_get_reg(M68K_REG_A0); save_param(&tmp32, 4);
tmp32 = s68k_get_reg(M68K_REG_A1); save_param(&tmp32, 4);
tmp32 = s68k_get_reg(M68K_REG_A2); save_param(&tmp32, 4);
tmp32 = s68k_get_reg(M68K_REG_A3); save_param(&tmp32, 4);
tmp32 = s68k_get_reg(M68K_REG_A4); save_param(&tmp32, 4);
tmp32 = s68k_get_reg(M68K_REG_A5); save_param(&tmp32, 4);
tmp32 = s68k_get_reg(M68K_REG_A6); save_param(&tmp32, 4);
tmp32 = s68k_get_reg(M68K_REG_A7); save_param(&tmp32, 4);
tmp32 = s68k_get_reg(M68K_REG_PC); save_param(&tmp32, 4);
tmp16 = s68k_get_reg(M68K_REG_SR); save_param(&tmp16, 2);
tmp32 = s68k_get_reg(M68K_REG_USP); save_param(&tmp32, 4);
tmp32 = s68k_get_reg(M68K_REG_ISP); save_param(&tmp32, 4);
if (scd.cartridge.boot)
{
bufferptr += md_cart_context_save(&state[bufferptr]);
}
save_param(scd.bram, 0x2000);
if (scd.cartridge.id)
save_param(scd.cartridge.area, scd.cartridge.mask + 1);
return bufferptr;
}
int scd_context_load(uint8 *state)
{
int i;
uint16 tmp16;
uint32 tmp32;
int bufferptr = 0;
load_param(scd.regs, sizeof(scd.regs));
load_param(&scd.cycles, sizeof(scd.cycles));
load_param(&scd.timer, sizeof(scd.timer));
load_param(&scd.pending, sizeof(scd.pending));
load_param(&scd.dmna, sizeof(scd.dmna));
bufferptr += gfx_context_load(&state[bufferptr]);
bufferptr += cdc_context_load(&state[bufferptr]);
bufferptr += cdd_context_load(&state[bufferptr]);
bufferptr += pcm_context_load(&state[bufferptr]);
load_param(scd.prg_ram, sizeof(scd.prg_ram));
m68k.memory_map[scd.cartridge.boot + 0x02].base = scd.prg_ram + ((scd.regs[0x03>>1].byte.l & 0xc0) << 11);
m68k.memory_map[scd.cartridge.boot + 0x03].base = m68k.memory_map[scd.cartridge.boot + 0x02].base + 0x10000;
if (scd.regs[0x03>>1].byte.l & 0x04)
{
load_param(scd.word_ram, sizeof(scd.word_ram));
if (scd.regs[0x03>>1].byte.l & 0x01)
{
for (i=scd.cartridge.boot+0x20; i<scd.cartridge.boot+0x22; i++)
{
m68k.memory_map[i].base = scd.word_ram[1] + ((i & 0x01) << 16);
}
for (i=scd.cartridge.boot+0x22; i<scd.cartridge.boot+0x24; i++)
{
m68k.memory_map[i].read8 = cell_ram_1_read8;
m68k.memory_map[i].read16 = cell_ram_1_read16;
m68k.memory_map[i].write8 = cell_ram_1_write8;
m68k.memory_map[i].write16 = cell_ram_1_write16;
zbank_memory_map[i].read = cell_ram_1_read8;
zbank_memory_map[i].write = cell_ram_1_write8;
}
for (i=0x08; i<0x0c; i++)
{
s68k.memory_map[i].read8 = dot_ram_0_read8;
s68k.memory_map[i].read16 = dot_ram_0_read16;
s68k.memory_map[i].write8 = dot_ram_0_write8;
s68k.memory_map[i].write16 = dot_ram_0_write16;
}
for (i=0x0c; i<0x0e; i++)
{
s68k.memory_map[i].base = scd.word_ram[0] + ((i & 0x01) << 16);
s68k.memory_map[i].read8 = NULL;
s68k.memory_map[i].read16 = NULL;
s68k.memory_map[i].write8 = NULL;
s68k.memory_map[i].write16 = NULL;
}
}
else
{
for (i=scd.cartridge.boot+0x20; i<scd.cartridge.boot+0x22; i++)
{
m68k.memory_map[i].base = scd.word_ram[0] + ((i & 0x01) << 16);
}
for (i=scd.cartridge.boot+0x22; i<scd.cartridge.boot+0x24; i++)
{
m68k.memory_map[i].read8 = cell_ram_0_read8;
m68k.memory_map[i].read16 = cell_ram_0_read16;
m68k.memory_map[i].write8 = cell_ram_0_write8;
m68k.memory_map[i].write16 = cell_ram_0_write16;
zbank_memory_map[i].read = cell_ram_0_read8;
zbank_memory_map[i].write = cell_ram_0_write8;
}
for (i=0x08; i<0x0c; i++)
{
s68k.memory_map[i].read8 = dot_ram_1_read8;
s68k.memory_map[i].read16 = dot_ram_1_read16;
s68k.memory_map[i].write8 = dot_ram_1_write8;
s68k.memory_map[i].write16 = dot_ram_1_write16;
}
for (i=0x0c; i<0x0e; i++)
{
s68k.memory_map[i].base = scd.word_ram[1] + ((i & 0x01) << 16);
s68k.memory_map[i].read8 = NULL;
s68k.memory_map[i].read16 = NULL;
s68k.memory_map[i].write8 = NULL;
s68k.memory_map[i].write16 = NULL;
}
}
}
else
{
load_param(scd.word_ram_2M, sizeof(scd.word_ram_2M));
for (i=scd.cartridge.boot+0x20; i<scd.cartridge.boot+0x24; i++)
{
m68k.memory_map[i].base = scd.word_ram_2M + ((i & 0x03) << 16);
m68k.memory_map[i].read8 = NULL;
m68k.memory_map[i].read16 = NULL;
m68k.memory_map[i].write8 = NULL;
m68k.memory_map[i].write16 = NULL;
zbank_memory_map[i].read = NULL;
zbank_memory_map[i].write = NULL;
}
for (i=0x08; i<0x0c; i++)
{
s68k.memory_map[i].read8 = NULL;
s68k.memory_map[i].read16 = NULL;
s68k.memory_map[i].write8 = NULL;
s68k.memory_map[i].write16 = NULL;
}
for (i=0x0c; i<0x0e; i++)
{
s68k.memory_map[i].read8 = s68k_read_bus_8;
s68k.memory_map[i].read16 = s68k_read_bus_16;
s68k.memory_map[i].write8 = s68k_unused_8_w;
s68k.memory_map[i].write16 = s68k_unused_16_w;
}
}
load_param(&m68k.poll, sizeof(m68k.poll));
load_param(&s68k.poll, sizeof(s68k.poll));
load_param(&tmp16, 2);
*(uint16 *)(m68k.memory_map[scd.cartridge.boot].base + 0x72) = tmp16;
load_param(&s68k.cycles, sizeof(s68k.cycles));
load_param(&s68k.int_level, sizeof(s68k.int_level));
load_param(&s68k.stopped, sizeof(s68k.stopped));
load_param(&tmp32, 4); s68k_set_reg(M68K_REG_D0, tmp32);
load_param(&tmp32, 4); s68k_set_reg(M68K_REG_D1, tmp32);
load_param(&tmp32, 4); s68k_set_reg(M68K_REG_D2, tmp32);
load_param(&tmp32, 4); s68k_set_reg(M68K_REG_D3, tmp32);
load_param(&tmp32, 4); s68k_set_reg(M68K_REG_D4, tmp32);
load_param(&tmp32, 4); s68k_set_reg(M68K_REG_D5, tmp32);
load_param(&tmp32, 4); s68k_set_reg(M68K_REG_D6, tmp32);
load_param(&tmp32, 4); s68k_set_reg(M68K_REG_D7, tmp32);
load_param(&tmp32, 4); s68k_set_reg(M68K_REG_A0, tmp32);
load_param(&tmp32, 4); s68k_set_reg(M68K_REG_A1, tmp32);
load_param(&tmp32, 4); s68k_set_reg(M68K_REG_A2, tmp32);
load_param(&tmp32, 4); s68k_set_reg(M68K_REG_A3, tmp32);
load_param(&tmp32, 4); s68k_set_reg(M68K_REG_A4, tmp32);
load_param(&tmp32, 4); s68k_set_reg(M68K_REG_A5, tmp32);
load_param(&tmp32, 4); s68k_set_reg(M68K_REG_A6, tmp32);
load_param(&tmp32, 4); s68k_set_reg(M68K_REG_A7, tmp32);
load_param(&tmp32, 4); s68k_set_reg(M68K_REG_PC, tmp32);
load_param(&tmp16, 2); s68k_set_reg(M68K_REG_SR, tmp16);
load_param(&tmp32, 4); s68k_set_reg(M68K_REG_USP,tmp32);
load_param(&tmp32, 4); s68k_set_reg(M68K_REG_ISP,tmp32);
if (scd.cartridge.boot)
{
bufferptr += md_cart_context_load(&state[bufferptr]);
}
load_param(scd.bram, 0x2000);
if (scd.cartridge.id)
load_param(scd.cartridge.area, scd.cartridge.mask + 1);
return bufferptr;
}
int scd_68k_irq_ack(int level)
{
#ifdef LOG_SCD
error("INT ack level %d (%X)\n", level, s68k.pc);
#endif
#if 0
if (level != 5)
#endif
{
scd.pending &= ~(1 << level);
if (level == 2)
{
scd.regs[0x00].byte.h &= ~0x01;
}
s68k_update_irq((scd.pending & scd.regs[0x32>>1].byte.l) >> 1);
}
return M68K_INT_ACK_AUTOVECTOR;
}