mirror of
https://github.com/libretro/Genesis-Plus-GX.git
synced 2024-11-27 18:40:42 +00:00
565 lines
18 KiB
C
565 lines
18 KiB
C
/***************************************************************************************
|
|
* Genesis Plus
|
|
* Internal Hardware & Bus controllers
|
|
*
|
|
* Support for SG-1000, Mark-III, Master System, Game Gear, Mega Drive & Mega CD hardware
|
|
*
|
|
* Copyright (C) 1998-2003 Charles Mac Donald (original code)
|
|
* Copyright (C) 2007-2018 Eke-Eke (Genesis Plus GX)
|
|
*
|
|
* Redistribution and use of this code or any derivative works are permitted
|
|
* provided that the following conditions are met:
|
|
*
|
|
* - Redistributions may not be sold, nor may they be used in a commercial
|
|
* product or activity.
|
|
*
|
|
* - Redistributions that are modified from the original source must include the
|
|
* complete source code, including the source code for all components used by a
|
|
* binary built from the modified sources. However, as a special exception, the
|
|
* source code distributed need not include anything that is normally distributed
|
|
* (in either source or binary form) with the major components (compiler, kernel,
|
|
* and so on) of the operating system on which the executable runs, unless that
|
|
* component itself accompanies the executable.
|
|
*
|
|
* - Redistributions must reproduce the above copyright notice, this list of
|
|
* conditions and the following disclaimer in the documentation and/or other
|
|
* materials provided with the distribution.
|
|
*
|
|
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
|
|
* AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
|
|
* IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
|
|
* ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE
|
|
* LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
|
|
* CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
|
|
* SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
|
|
* INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
|
|
* CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
|
|
* ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
|
|
* POSSIBILITY OF SUCH DAMAGE.
|
|
*
|
|
****************************************************************************************/
|
|
|
|
#include "shared.h"
|
|
|
|
#ifdef USE_DYNAMIC_ALLOC
|
|
external_t *ext;
|
|
#else /* External Hardware (Cartridge, CD unit, ...) */
|
|
external_t ext;
|
|
#endif
|
|
uint8 boot_rom[0x800]; /* Genesis BOOT ROM */
|
|
uint8 work_ram[0x10000]; /* 68K RAM */
|
|
uint8 zram[0x2000]; /* Z80 RAM */
|
|
uint32 zbank; /* Z80 bank window address */
|
|
uint8 zstate; /* Z80 bus state (d0 = BUSACK, d1 = /RESET) */
|
|
uint8 pico_current; /* PICO current page */
|
|
|
|
static uint8 tmss[4]; /* TMSS security register */
|
|
|
|
/*--------------------------------------------------------------------------*/
|
|
/* Init, reset, shutdown functions */
|
|
/*--------------------------------------------------------------------------*/
|
|
|
|
void gen_init(void)
|
|
{
|
|
int i;
|
|
|
|
/* initialize Z80 */
|
|
z80_init(0,z80_irq_callback);
|
|
|
|
/* 8-bit / 16-bit modes */
|
|
if ((system_hw & SYSTEM_PBC) == SYSTEM_MD)
|
|
{
|
|
/* initialize main 68k */
|
|
m68k_init();
|
|
m68k.aerr_enabled = config.addr_error;
|
|
|
|
/* initialize main 68k memory map */
|
|
|
|
/* $800000-$DFFFFF : illegal access by default */
|
|
for (i=0x80; i<0xe0; i++)
|
|
{
|
|
m68k.memory_map[i].base = work_ram; /* for VDP DMA */
|
|
m68k.memory_map[i].read8 = m68k_lockup_r_8;
|
|
m68k.memory_map[i].read16 = m68k_lockup_r_16;
|
|
m68k.memory_map[i].write8 = m68k_lockup_w_8;
|
|
m68k.memory_map[i].write16 = m68k_lockup_w_16;
|
|
zbank_memory_map[i].read = zbank_lockup_r;
|
|
zbank_memory_map[i].write = zbank_lockup_w;
|
|
}
|
|
|
|
/* $C0xxxx, $C8xxxx, $D0xxxx, $D8xxxx : VDP ports */
|
|
for (i=0xc0; i<0xe0; i+=8)
|
|
{
|
|
m68k.memory_map[i].read8 = vdp_read_byte;
|
|
m68k.memory_map[i].read16 = vdp_read_word;
|
|
m68k.memory_map[i].write8 = vdp_write_byte;
|
|
m68k.memory_map[i].write16 = vdp_write_word;
|
|
zbank_memory_map[i].read = zbank_read_vdp;
|
|
zbank_memory_map[i].write = zbank_write_vdp;
|
|
}
|
|
|
|
/* $E00000-$FFFFFF : Work RAM (64k) */
|
|
for (i=0xe0; i<0x100; i++)
|
|
{
|
|
m68k.memory_map[i].base = work_ram;
|
|
m68k.memory_map[i].read8 = NULL;
|
|
m68k.memory_map[i].read16 = NULL;
|
|
m68k.memory_map[i].write8 = NULL;
|
|
m68k.memory_map[i].write16 = NULL;
|
|
|
|
/* Z80 can ONLY write to 68k RAM, not read it */
|
|
zbank_memory_map[i].read = zbank_unused_r;
|
|
zbank_memory_map[i].write = NULL;
|
|
}
|
|
|
|
if (system_hw == SYSTEM_PICO)
|
|
{
|
|
/* additional registers mapped to $800000-$80FFFF */
|
|
m68k.memory_map[0x80].read8 = pico_read_byte;
|
|
m68k.memory_map[0x80].read16 = pico_read_word;
|
|
m68k.memory_map[0x80].write8 = m68k_unused_8_w;
|
|
m68k.memory_map[0x80].write16 = m68k_unused_16_w;
|
|
|
|
/* there is no I/O area (Notaz) */
|
|
m68k.memory_map[0xa1].read8 = m68k_read_bus_8;
|
|
m68k.memory_map[0xa1].read16 = m68k_read_bus_16;
|
|
m68k.memory_map[0xa1].write8 = m68k_unused_8_w;
|
|
m68k.memory_map[0xa1].write16 = m68k_unused_16_w;
|
|
|
|
/* initialize page index (closed) */
|
|
pico_current = 0;
|
|
}
|
|
else
|
|
{
|
|
/* $A10000-$A1FFFF : I/O & Control registers */
|
|
m68k.memory_map[0xa1].read8 = ctrl_io_read_byte;
|
|
m68k.memory_map[0xa1].read16 = ctrl_io_read_word;
|
|
m68k.memory_map[0xa1].write8 = ctrl_io_write_byte;
|
|
m68k.memory_map[0xa1].write16 = ctrl_io_write_word;
|
|
zbank_memory_map[0xa1].read = zbank_read_ctrl_io;
|
|
zbank_memory_map[0xa1].write = zbank_write_ctrl_io;
|
|
|
|
/* initialize Z80 memory map */
|
|
/* $0000-$3FFF is mapped to Z80 RAM (8K mirrored) */
|
|
/* $4000-$FFFF is mapped to hardware but Z80 PC should never point there */
|
|
for (i=0; i<64; i++)
|
|
{
|
|
z80_readmap[i] = &zram[(i & 7) << 10];
|
|
}
|
|
|
|
/* initialize Z80 memory handlers */
|
|
z80_writemem = z80_memory_w;
|
|
z80_readmem = z80_memory_r;
|
|
|
|
/* initialize Z80 port handlers */
|
|
z80_writeport = z80_unused_port_w;
|
|
z80_readport = z80_unused_port_r;
|
|
}
|
|
|
|
/* $000000-$7FFFFF : external hardware area */
|
|
if (system_hw == SYSTEM_MCD)
|
|
{
|
|
/* initialize SUB-CPU */
|
|
s68k_init();
|
|
|
|
/* initialize CD hardware */
|
|
scd_init();
|
|
}
|
|
else
|
|
{
|
|
/* Cartridge hardware */
|
|
md_cart_init();
|
|
}
|
|
}
|
|
else
|
|
{
|
|
/* initialize cartridge hardware & Z80 memory handlers */
|
|
sms_cart_init();
|
|
|
|
/* initialize Z80 ports handlers */
|
|
switch (system_hw)
|
|
{
|
|
/* Master System compatibility mode */
|
|
case SYSTEM_PBC:
|
|
{
|
|
z80_writeport = z80_md_port_w;
|
|
z80_readport = z80_md_port_r;
|
|
break;
|
|
}
|
|
|
|
/* Game Gear hardware */
|
|
case SYSTEM_GG:
|
|
case SYSTEM_GGMS:
|
|
{
|
|
/* initialize cartridge hardware & Z80 memory handlers */
|
|
sms_cart_init();
|
|
|
|
/* initialize Z80 ports handlers */
|
|
z80_writeport = z80_gg_port_w;
|
|
z80_readport = z80_gg_port_r;
|
|
break;
|
|
}
|
|
|
|
/* Master System hardware */
|
|
case SYSTEM_SMS:
|
|
case SYSTEM_SMS2:
|
|
{
|
|
z80_writeport = z80_ms_port_w;
|
|
z80_readport = z80_ms_port_r;
|
|
break;
|
|
}
|
|
|
|
/* Mark-III hardware */
|
|
case SYSTEM_MARKIII:
|
|
{
|
|
z80_writeport = z80_m3_port_w;
|
|
z80_readport = z80_m3_port_r;
|
|
break;
|
|
}
|
|
|
|
/* SG-1000 hardware */
|
|
case SYSTEM_SG:
|
|
case SYSTEM_SGII:
|
|
{
|
|
z80_writeport = z80_sg_port_w;
|
|
z80_readport = z80_sg_port_r;
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
void gen_reset(int hard_reset)
|
|
{
|
|
/* System Reset */
|
|
if (hard_reset)
|
|
{
|
|
/* On hard reset, 68k CPU always starts at the same point in VDP frame */
|
|
/* Tests performed on VA4 PAL MD1 showed that the first HVC value read */
|
|
/* with 'move.w #0x8104,0xC00004' , 'move.w 0xC00008,%d0' sequence was */
|
|
/* 0x9F21 in 60HZ mode (0x9F00 if Mode 5 is not enabled by first MOVE) */
|
|
/* 0x8421 in 50HZ mode (0x8400 if Mode 5 is not enabled by first MOVE) */
|
|
/* Same value is returned on every power ON, indicating VDP is always */
|
|
/* starting at the same fixed point in frame (probably at the start of */
|
|
/* VSYNC and HSYNC) while 68k /VRES line remains asserted a fixed time */
|
|
/* after /SRES line has been released (13 msec approx). The difference */
|
|
/* between PAL & NTSC is caused by the top border area being 27 lines */
|
|
/* larger in PAL mode than in NTSC mode. CPU cycle counter is adjusted */
|
|
/* to match these results (taking in account emulated frame is started */
|
|
/* on line 192 */
|
|
m68k.cycles = ((lines_per_frame - 192 + 159 - (27 * vdp_pal)) * MCYCLES_PER_LINE) + 1004;
|
|
|
|
/* clear RAM (on real hardware, RAM values are random / undetermined on Power ON) */
|
|
memset(work_ram, 0x00, sizeof (work_ram));
|
|
memset(zram, 0x00, sizeof (zram));
|
|
}
|
|
else
|
|
{
|
|
/* when RESET button is pressed, 68k could be anywhere in VDP frame (Bonkers, Eternal Champions, X-Men 2) */
|
|
m68k.cycles = (uint32)((MCYCLES_PER_LINE * lines_per_frame) * ((double)rand() / (double)RAND_MAX));
|
|
|
|
/* reset YM2612 (on hard reset, this is done by sound_reset) */
|
|
fm_reset(0);
|
|
}
|
|
|
|
/* 68k M-cycles should be a multiple of 7 */
|
|
m68k.cycles = (m68k.cycles / 7) * 7;
|
|
|
|
/* Z80 M-cycles should be a multiple of 15 */
|
|
Z80.cycles = (m68k.cycles / 15) * 15;
|
|
|
|
/* 8-bit / 16-bit modes */
|
|
if ((system_hw & SYSTEM_PBC) == SYSTEM_MD)
|
|
{
|
|
if (system_hw == SYSTEM_MCD)
|
|
{
|
|
/* FRES is only asserted on Power ON */
|
|
if (hard_reset)
|
|
{
|
|
/* reset CD hardware */
|
|
scd_reset(1);
|
|
}
|
|
}
|
|
|
|
/* reset MD cartridge hardware */
|
|
md_cart_reset(hard_reset);
|
|
|
|
/* Z80 bus is released & Z80 is reseted */
|
|
m68k.memory_map[0xa0].read8 = m68k_read_bus_8;
|
|
m68k.memory_map[0xa0].read16 = m68k_read_bus_16;
|
|
m68k.memory_map[0xa0].write8 = m68k_unused_8_w;
|
|
m68k.memory_map[0xa0].write16 = m68k_unused_16_w;
|
|
zstate = 0;
|
|
|
|
/* assume default bank is $000000-$007FFF */
|
|
zbank = 0;
|
|
|
|
/* TMSS support */
|
|
if ((config.bios & 1) && (system_hw == SYSTEM_MD) && hard_reset)
|
|
{
|
|
int i;
|
|
|
|
/* clear TMSS register */
|
|
memset(tmss, 0x00, sizeof(tmss));
|
|
|
|
/* VDP access is locked by default */
|
|
for (i=0xc0; i<0xe0; i+=8)
|
|
{
|
|
m68k.memory_map[i].read8 = m68k_lockup_r_8;
|
|
m68k.memory_map[i].read16 = m68k_lockup_r_16;
|
|
m68k.memory_map[i].write8 = m68k_lockup_w_8;
|
|
m68k.memory_map[i].write16 = m68k_lockup_w_16;
|
|
zbank_memory_map[i].read = zbank_lockup_r;
|
|
zbank_memory_map[i].write = zbank_lockup_w;
|
|
}
|
|
|
|
/* check if BOOT ROM is loaded */
|
|
if (system_bios & SYSTEM_MD)
|
|
{
|
|
/* save default cartridge slot mapping */
|
|
cart.base = m68k.memory_map[0].base;
|
|
|
|
/* BOOT ROM is mapped at $000000-$0007FF */
|
|
m68k.memory_map[0].base = boot_rom;
|
|
}
|
|
}
|
|
|
|
/* reset MAIN-CPU */
|
|
m68k_pulse_reset();
|
|
}
|
|
else
|
|
{
|
|
/* RAM state at power-on is undefined on some systems */
|
|
if ((system_hw == SYSTEM_MARKIII) || ((system_hw & SYSTEM_SMS) && (region_code == REGION_JAPAN_NTSC)))
|
|
{
|
|
/* some korean games rely on RAM to be initialized with values different from $00 or $ff */
|
|
memset(work_ram, 0xf0, sizeof(work_ram));
|
|
}
|
|
|
|
/* reset cartridge hardware */
|
|
sms_cart_reset();
|
|
|
|
/* halt 68k (/VRES is forced low) */
|
|
m68k_pulse_halt();
|
|
}
|
|
|
|
/* reset Z80 */
|
|
z80_reset();
|
|
|
|
/* some Z80 registers need to be initialized on Power ON */
|
|
if (hard_reset)
|
|
{
|
|
/* Power Base Converter specific */
|
|
if (system_hw == SYSTEM_PBC)
|
|
{
|
|
/* startup code logic (verified on real hardware): */
|
|
/* 21 01 E1 : LD HL, $E101
|
|
25 -- -- : DEC H
|
|
F9 -- -- : LD SP,HL
|
|
C7 -- -- : RST $00
|
|
01 01 -- : LD BC, $xx01
|
|
*/
|
|
Z80.hl.w.l = 0xE001;
|
|
Z80.sp.w.l = 0xDFFF;
|
|
Z80.r = 4;
|
|
}
|
|
|
|
/* Master System & Game Gear specific */
|
|
else if (system_hw & (SYSTEM_SMS | SYSTEM_GG))
|
|
{
|
|
/* check if BIOS is not being used */
|
|
if ((!(config.bios & 1) || !(system_bios & (SYSTEM_SMS | SYSTEM_GG))))
|
|
{
|
|
/* a few Master System (Ace of Aces, Shadow Dancer) & Game Gear (Ecco the Dolphin, Evander Holyfield Real Deal Boxing) games crash if SP is not properly initialized */
|
|
Z80.sp.w.l = 0xDFF0;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
/*-----------------------------------------------------------------------*/
|
|
/* OS ROM / TMSS register control functions (Genesis mode) */
|
|
/*-----------------------------------------------------------------------*/
|
|
|
|
void gen_tmss_w(unsigned int offset, unsigned int data)
|
|
{
|
|
int i;
|
|
|
|
/* write TMSS register */
|
|
WRITE_WORD(tmss, offset, data);
|
|
|
|
/* VDP requires "SEGA" value to be written in TMSS register */
|
|
if (memcmp((char *)tmss, "SEGA", 4) == 0)
|
|
{
|
|
for (i=0xc0; i<0xe0; i+=8)
|
|
{
|
|
m68k.memory_map[i].read8 = vdp_read_byte;
|
|
m68k.memory_map[i].read16 = vdp_read_word;
|
|
m68k.memory_map[i].write8 = vdp_write_byte;
|
|
m68k.memory_map[i].write16 = vdp_write_word;
|
|
zbank_memory_map[i].read = zbank_read_vdp;
|
|
zbank_memory_map[i].write = zbank_write_vdp;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
for (i=0xc0; i<0xe0; i+=8)
|
|
{
|
|
m68k.memory_map[i].read8 = m68k_lockup_r_8;
|
|
m68k.memory_map[i].read16 = m68k_lockup_r_16;
|
|
m68k.memory_map[i].write8 = m68k_lockup_w_8;
|
|
m68k.memory_map[i].write16 = m68k_lockup_w_16;
|
|
zbank_memory_map[i].read = zbank_lockup_r;
|
|
zbank_memory_map[i].write = zbank_lockup_w;
|
|
}
|
|
}
|
|
}
|
|
|
|
void gen_bankswitch_w(unsigned int data)
|
|
{
|
|
/* check if BOOT ROM is loaded */
|
|
if (system_bios & SYSTEM_MD)
|
|
{
|
|
if (data & 1)
|
|
{
|
|
/* enable cartridge ROM */
|
|
m68k.memory_map[0].base = cart.base;
|
|
}
|
|
else
|
|
{
|
|
/* enable internal BOOT ROM */
|
|
m68k.memory_map[0].base = boot_rom;
|
|
}
|
|
}
|
|
}
|
|
|
|
unsigned int gen_bankswitch_r(void)
|
|
{
|
|
/* check if BOOT ROM is loaded */
|
|
if (system_bios & SYSTEM_MD)
|
|
{
|
|
return (m68k.memory_map[0].base == cart.base);
|
|
}
|
|
|
|
return 0xff;
|
|
}
|
|
|
|
|
|
/*-----------------------------------------------------------------------*/
|
|
/* Z80 Bus controller chip functions (Genesis mode) */
|
|
/* ----------------------------------------------------------------------*/
|
|
|
|
void gen_zbusreq_w(unsigned int data, unsigned int cycles)
|
|
{
|
|
if (data) /* !ZBUSREQ asserted */
|
|
{
|
|
/* check if Z80 is going to be stopped */
|
|
if (zstate == 1)
|
|
{
|
|
/* resynchronize with 68k */
|
|
z80_run(cycles);
|
|
|
|
/* enable 68k access to Z80 bus */
|
|
m68k.memory_map[0xa0].read8 = z80_read_byte;
|
|
m68k.memory_map[0xa0].read16 = z80_read_word;
|
|
m68k.memory_map[0xa0].write8 = z80_write_byte;
|
|
m68k.memory_map[0xa0].write16 = z80_write_word;
|
|
}
|
|
|
|
/* update Z80 bus status */
|
|
zstate |= 2;
|
|
}
|
|
else /* !ZBUSREQ released */
|
|
{
|
|
/* check if Z80 is going to be restarted */
|
|
if (zstate == 3)
|
|
{
|
|
/* resynchronize with 68k (Z80 cycles should remain a multiple of 15 MClocks) */
|
|
Z80.cycles = ((cycles + 14) / 15) * 15;
|
|
|
|
/* disable 68k access to Z80 bus */
|
|
m68k.memory_map[0xa0].read8 = m68k_read_bus_8;
|
|
m68k.memory_map[0xa0].read16 = m68k_read_bus_16;
|
|
m68k.memory_map[0xa0].write8 = m68k_unused_8_w;
|
|
m68k.memory_map[0xa0].write16 = m68k_unused_16_w;
|
|
}
|
|
|
|
/* update Z80 bus status */
|
|
zstate &= 1;
|
|
}
|
|
}
|
|
|
|
void gen_zreset_w(unsigned int data, unsigned int cycles)
|
|
{
|
|
if (data) /* !ZRESET released */
|
|
{
|
|
/* check if Z80 is going to be restarted */
|
|
if (zstate == 0)
|
|
{
|
|
/* resynchronize with 68k (Z80 cycles should remain a multiple of 15 MClocks) */
|
|
Z80.cycles = ((cycles + 14) / 15) * 15;
|
|
|
|
/* reset Z80 & YM2612 */
|
|
z80_reset();
|
|
fm_reset(cycles);
|
|
}
|
|
|
|
/* check if 68k access to Z80 bus is granted */
|
|
else if (zstate == 2)
|
|
{
|
|
/* enable 68k access to Z80 bus */
|
|
m68k.memory_map[0xa0].read8 = z80_read_byte;
|
|
m68k.memory_map[0xa0].read16 = z80_read_word;
|
|
m68k.memory_map[0xa0].write8 = z80_write_byte;
|
|
m68k.memory_map[0xa0].write16 = z80_write_word;
|
|
|
|
/* reset Z80 & YM2612 */
|
|
z80_reset();
|
|
fm_reset(cycles);
|
|
}
|
|
|
|
/* update Z80 bus status */
|
|
zstate |= 1;
|
|
}
|
|
else /* !ZRESET asserted */
|
|
{
|
|
/* check if Z80 is going to be stopped */
|
|
if (zstate == 1)
|
|
{
|
|
/* resynchronize with 68k */
|
|
z80_run(cycles);
|
|
}
|
|
|
|
/* check if 68k had access to Z80 bus */
|
|
else if (zstate == 3)
|
|
{
|
|
/* disable 68k access to Z80 bus */
|
|
m68k.memory_map[0xa0].read8 = m68k_read_bus_8;
|
|
m68k.memory_map[0xa0].read16 = m68k_read_bus_16;
|
|
m68k.memory_map[0xa0].write8 = m68k_unused_8_w;
|
|
m68k.memory_map[0xa0].write16 = m68k_unused_16_w;
|
|
}
|
|
|
|
/* stop YM2612 */
|
|
fm_reset(cycles);
|
|
|
|
/* update Z80 bus status */
|
|
zstate &= 2;
|
|
}
|
|
}
|
|
|
|
void gen_zbank_w (unsigned int data)
|
|
{
|
|
zbank = ((zbank >> 1) | ((data & 1) << 23)) & 0xFF8000;
|
|
}
|
|
|
|
|
|
/*-----------------------------------------------------------------------*/
|
|
/* Z80 interrupt callback */
|
|
/* ----------------------------------------------------------------------*/
|
|
|
|
int z80_irq_callback (int param)
|
|
{
|
|
return -1;
|
|
}
|