/***************************************************************************************1* Genesis Plus2* Internal Hardware & Bus controllers3*4* Support for SG-1000, Mark-III, Master System, Game Gear, Mega Drive & Mega CD hardware5*6* Copyright (C) 1998, 1999, 2000, 2001, 2002, 2003 Charles Mac Donald (original code)7* Copyright (C) 2007-2013 Eke-Eke (Genesis Plus GX)8*9* Redistribution and use of this code or any derivative works are permitted10* provided that the following conditions are met:11*12* - Redistributions may not be sold, nor may they be used in a commercial13* product or activity.14*15* - Redistributions that are modified from the original source must include the16* complete source code, including the source code for all components used by a17* binary built from the modified sources. However, as a special exception, the18* source code distributed need not include anything that is normally distributed19* (in either source or binary form) with the major components (compiler, kernel,20* and so on) of the operating system on which the executable runs, unless that21* component itself accompanies the executable.22*23* - Redistributions must reproduce the above copyright notice, this list of24* conditions and the following disclaimer in the documentation and/or other25* materials provided with the distribution.26*27* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"28* AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE29* IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE30* ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE31* LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR32* CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF33* SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS34* INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN35* CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)36* ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE37* POSSIBILITY OF SUCH DAMAGE.38*39****************************************************************************************/4041#include "shared.h"4243external_t ext; /* External Hardware (Cartridge, CD unit, ...) */44uint8 boot_rom[0x800]; /* Genesis BOOT ROM */45uint8 work_ram[0x10000]; /* 68K RAM */46uint8 zram[0x2000]; /* Z80 RAM */47uint32 zbank; /* Z80 bank window address */48uint8 zstate; /* Z80 bus state (d0 = BUSACK, d1 = /RESET) */49uint8 pico_current; /* PICO current page */5051uint8 tmss[4]; /* TMSS security register */5253/*--------------------------------------------------------------------------*/54/* Init, reset, shutdown functions */55/*--------------------------------------------------------------------------*/5657void gen_init(void)58{59int i;6061/* initialize Z80 */62z80_init(0,z80_irq_callback);6364/* 8-bit / 16-bit modes */65if ((system_hw & SYSTEM_PBC) == SYSTEM_MD)66{67/* initialize main 68k */68m68k_init();69m68k.aerr_enabled = config.addr_error;7071/* initialize main 68k memory map */7273/* $800000-$DFFFFF : illegal access by default */74for (i=0x80; i<0xe0; i++)75{76m68k.memory_map[i].base = work_ram; /* for VDP DMA */77m68k.memory_map[i].read8 = m68k_lockup_r_8;78m68k.memory_map[i].read16 = m68k_lockup_r_16;79m68k.memory_map[i].write8 = m68k_lockup_w_8;80m68k.memory_map[i].write16 = m68k_lockup_w_16;81zbank_memory_map[i].read = zbank_lockup_r;82zbank_memory_map[i].write = zbank_lockup_w;83}8485/* $C0xxxx, $C8xxxx, $D0xxxx, $D8xxxx : VDP ports */86for (i=0xc0; i<0xe0; i+=8)87{88m68k.memory_map[i].read8 = vdp_read_byte;89m68k.memory_map[i].read16 = vdp_read_word;90m68k.memory_map[i].write8 = vdp_write_byte;91m68k.memory_map[i].write16 = vdp_write_word;92zbank_memory_map[i].read = zbank_read_vdp;93zbank_memory_map[i].write = zbank_write_vdp;94}9596/* $E00000-$FFFFFF : Work RAM (64k) */97for (i=0xe0; i<0x100; i++)98{99m68k.memory_map[i].base = work_ram;100m68k.memory_map[i].read8 = NULL;101m68k.memory_map[i].read16 = NULL;102m68k.memory_map[i].write8 = NULL;103m68k.memory_map[i].write16 = NULL;104105/* Z80 can ONLY write to 68k RAM, not read it */106zbank_memory_map[i].read = zbank_unused_r;107zbank_memory_map[i].write = NULL;108}109110if (system_hw == SYSTEM_PICO)111{112/* additional registers mapped to $800000-$80FFFF */113m68k.memory_map[0x80].read8 = pico_read_byte;114m68k.memory_map[0x80].read16 = pico_read_word;115m68k.memory_map[0x80].write8 = m68k_unused_8_w;116m68k.memory_map[0x80].write16 = m68k_unused_16_w;117118/* there is no I/O area (Notaz) */119m68k.memory_map[0xa1].read8 = m68k_read_bus_8;120m68k.memory_map[0xa1].read16 = m68k_read_bus_16;121m68k.memory_map[0xa1].write8 = m68k_unused_8_w;122m68k.memory_map[0xa1].write16 = m68k_unused_16_w;123124/* initialize page index (closed) */125pico_current = 0;126}127else128{129/* $A10000-$A1FFFF : I/O & Control registers */130m68k.memory_map[0xa1].read8 = ctrl_io_read_byte;131m68k.memory_map[0xa1].read16 = ctrl_io_read_word;132m68k.memory_map[0xa1].write8 = ctrl_io_write_byte;133m68k.memory_map[0xa1].write16 = ctrl_io_write_word;134zbank_memory_map[0xa1].read = zbank_read_ctrl_io;135zbank_memory_map[0xa1].write = zbank_write_ctrl_io;136137/* initialize Z80 memory map */138/* $0000-$3FFF is mapped to Z80 RAM (8K mirrored) */139/* $4000-$FFFF is mapped to hardware but Z80 PC should never point there */140for (i=0; i<64; i++)141{142z80_readmap[i] = &zram[(i & 7) << 10];143}144145/* initialize Z80 memory handlers */146z80_writemem = z80_memory_w;147z80_readmem = z80_memory_r;148149/* initialize Z80 port handlers */150z80_writeport = z80_unused_port_w;151z80_readport = z80_unused_port_r;152}153154/* $000000-$7FFFFF : external hardware area */155if (system_hw == SYSTEM_MCD)156{157/* initialize SUB-CPU */158s68k_init();159160/* initialize CD hardware */161scd_init();162}163else164{165/* Cartridge hardware */166md_cart_init();167}168}169else170{171/* initialize cartridge hardware & Z80 memory handlers */172sms_cart_init();173174/* initialize Z80 ports handlers */175switch (system_hw)176{177/* Master System compatibility mode */178case SYSTEM_PBC:179{180z80_writeport = z80_md_port_w;181z80_readport = z80_md_port_r;182break;183}184185/* Game Gear hardware */186case SYSTEM_GG:187case SYSTEM_GGMS:188{189/* initialize cartridge hardware & Z80 memory handlers */190sms_cart_init();191192/* initialize Z80 ports handlers */193z80_writeport = z80_gg_port_w;194z80_readport = z80_gg_port_r;195break;196}197198/* Master SYstem hardware */199case SYSTEM_SMS:200case SYSTEM_SMS2:201{202z80_writeport = z80_ms_port_w;203z80_readport = z80_ms_port_r;204break;205}206207/* Mark-III hardware */208case SYSTEM_MARKIII:209{210z80_writeport = z80_m3_port_w;211z80_readport = z80_m3_port_r;212break;213}214215/* SG-1000 hardware */216case SYSTEM_SG:217{218z80_writeport = z80_sg_port_w;219z80_readport = z80_sg_port_r;220break;221}222}223}224}225226void gen_reset(int hard_reset)227{228/* System Reset */229if (hard_reset)230{231/* clear RAM (TODO: use random bit patterns for all systems, like on real hardware) */232memset(work_ram, 0x00, sizeof (work_ram));233memset(zram, 0x00, sizeof (zram));234}235else236{237/* reset YM2612 (on hard reset, this is done by sound_reset) */238fm_reset(0);239}240241/* 68k & Z80 could be anywhere in VDP frame (Bonkers, Eternal Champions, X-Men 2) */242m68k.cycles = Z80.cycles = 0; //(uint32)((MCYCLES_PER_LINE * lines_per_frame) * ((double)rand() / (double)RAND_MAX));243244/* 68k cycles should be a multiple of 7 */245m68k.cycles = (m68k.cycles / 7) * 7;246247/* Z80 cycles should be a multiple of 15 */248Z80.cycles = (Z80.cycles / 15) * 15;249250/* 8-bit / 16-bit modes */251if ((system_hw & SYSTEM_PBC) == SYSTEM_MD)252{253if (system_hw == SYSTEM_MCD)254{255/* FRES is only asserted on Power ON */256if (hard_reset)257{258/* reset CD hardware */259scd_reset(1);260}261}262263/* reset MD cartridge hardware */264md_cart_reset(hard_reset);265266/* Z80 bus is released & Z80 is reseted */267m68k.memory_map[0xa0].read8 = m68k_read_bus_8;268m68k.memory_map[0xa0].read16 = m68k_read_bus_16;269m68k.memory_map[0xa0].write8 = m68k_unused_8_w;270m68k.memory_map[0xa0].write16 = m68k_unused_16_w;271zstate = 0;272273/* assume default bank is $000000-$007FFF */274zbank = 0;275276/* TMSS support */277if ((config.bios & 1) && (system_hw == SYSTEM_MD) && hard_reset)278{279int i;280281/* clear TMSS register */282memset(tmss, 0x00, sizeof(tmss));283284/* VDP access is locked by default */285for (i=0xc0; i<0xe0; i+=8)286{287m68k.memory_map[i].read8 = m68k_lockup_r_8;288m68k.memory_map[i].read16 = m68k_lockup_r_16;289m68k.memory_map[i].write8 = m68k_lockup_w_8;290m68k.memory_map[i].write16 = m68k_lockup_w_16;291zbank_memory_map[i].read = zbank_lockup_r;292zbank_memory_map[i].write = zbank_lockup_w;293}294295/* check if BOOT ROM is loaded */296if (system_bios & SYSTEM_MD)297{298/* save default cartridge slot mapping */299cart.base = m68k.memory_map[0].base;300301/* BOOT ROM is mapped at $000000-$0007FF */302m68k.memory_map[0].base = boot_rom;303}304}305306/* reset MAIN-CPU */307m68k_pulse_reset();308}309else310{311/* RAM state at power-on is undefined on some systems */312if ((system_hw == SYSTEM_MARKIII) || ((system_hw & SYSTEM_SMS) && (region_code == REGION_JAPAN_NTSC)))313{314/* some korean games rely on RAM to be initialized with values different from $00 or $ff */315memset(work_ram, 0xf0, sizeof(work_ram));316}317318/* reset cartridge hardware */319sms_cart_reset();320321/* halt 68k (/VRES is forced low) */322m68k_pulse_halt();323}324325/* reset Z80 */326z80_reset();327328/* some Z80 registers need to be initialized on Power ON */329if (hard_reset)330{331/* Power Base Converter specific */332if (system_hw == SYSTEM_PBC)333{334/* startup code logic (verified on real hardware): */335/* 21 01 E1 : LD HL, $E10133625 -- -- : DEC H337F9 -- -- : LD SP,HL338C7 -- -- : RST $0033901 01 -- : LD BC, $xx01340*/341Z80.hl.w.l = 0xE001;342Z80.sp.w.l = 0xDFFF;343Z80.r = 4;344}345346/* Master System specific (when BIOS is disabled) */347else if ((system_hw & SYSTEM_SMS) && (!(config.bios & 1) || !(system_bios & SYSTEM_SMS)))348{349/* usually done by BIOS & required by some SMS games that don't initialize SP */350Z80.sp.w.l = 0xDFFF;351}352}353}354355/*-----------------------------------------------------------------------*/356/* OS ROM / TMSS register control functions (Genesis mode) */357/*-----------------------------------------------------------------------*/358359void gen_tmss_w(unsigned int offset, unsigned int data)360{361int i;362363/* write TMSS register */364WRITE_WORD(tmss, offset, data);365366/* VDP requires "SEGA" value to be written in TMSS register */367if (memcmp((char *)tmss, "SEGA", 4) == 0)368{369for (i=0xc0; i<0xe0; i+=8)370{371m68k.memory_map[i].read8 = vdp_read_byte;372m68k.memory_map[i].read16 = vdp_read_word;373m68k.memory_map[i].write8 = vdp_write_byte;374m68k.memory_map[i].write16 = vdp_write_word;375zbank_memory_map[i].read = zbank_read_vdp;376zbank_memory_map[i].write = zbank_write_vdp;377}378}379else380{381for (i=0xc0; i<0xe0; i+=8)382{383m68k.memory_map[i].read8 = m68k_lockup_r_8;384m68k.memory_map[i].read16 = m68k_lockup_r_16;385m68k.memory_map[i].write8 = m68k_lockup_w_8;386m68k.memory_map[i].write16 = m68k_lockup_w_16;387zbank_memory_map[i].read = zbank_lockup_r;388zbank_memory_map[i].write = zbank_lockup_w;389}390}391}392393void gen_bankswitch_w(unsigned int data)394{395/* check if BOOT ROM is loaded */396if (system_bios & SYSTEM_MD)397{398if (data & 1)399{400/* enable cartridge ROM */401m68k.memory_map[0].base = cart.base;402}403else404{405/* enable internal BOOT ROM */406m68k.memory_map[0].base = boot_rom;407}408}409}410411unsigned int gen_bankswitch_r(void)412{413/* check if BOOT ROM is loaded */414if (system_bios & SYSTEM_MD)415{416return (m68k.memory_map[0].base == cart.base);417}418419return 0xff;420}421422423/*-----------------------------------------------------------------------*/424/* Z80 Bus controller chip functions (Genesis mode) */425/* ----------------------------------------------------------------------*/426427void gen_zbusreq_w(unsigned int data, unsigned int cycles)428{429if (data) /* !ZBUSREQ asserted */430{431/* check if Z80 is going to be stopped */432if (zstate == 1)433{434/* resynchronize with 68k */435z80_run(cycles);436437/* enable 68k access to Z80 bus */438m68k.memory_map[0xa0].read8 = z80_read_byte;439m68k.memory_map[0xa0].read16 = z80_read_word;440m68k.memory_map[0xa0].write8 = z80_write_byte;441m68k.memory_map[0xa0].write16 = z80_write_word;442}443444/* update Z80 bus status */445zstate |= 2;446}447else /* !ZBUSREQ released */448{449/* check if Z80 is going to be restarted */450if (zstate == 3)451{452/* resynchronize with 68k */453Z80.cycles = cycles;454455/* disable 68k access to Z80 bus */456m68k.memory_map[0xa0].read8 = m68k_read_bus_8;457m68k.memory_map[0xa0].read16 = m68k_read_bus_16;458m68k.memory_map[0xa0].write8 = m68k_unused_8_w;459m68k.memory_map[0xa0].write16 = m68k_unused_16_w;460}461462/* update Z80 bus status */463zstate &= 1;464}465}466467void gen_zreset_w(unsigned int data, unsigned int cycles)468{469if (data) /* !ZRESET released */470{471/* check if Z80 is going to be restarted */472if (zstate == 0)473{474/* resynchronize with 68k */475Z80.cycles = cycles;476477/* reset Z80 & YM2612 */478z80_reset();479fm_reset(cycles);480}481482/* check if 68k access to Z80 bus is granted */483else if (zstate == 2)484{485/* enable 68k access to Z80 bus */486m68k.memory_map[0xa0].read8 = z80_read_byte;487m68k.memory_map[0xa0].read16 = z80_read_word;488m68k.memory_map[0xa0].write8 = z80_write_byte;489m68k.memory_map[0xa0].write16 = z80_write_word;490491/* reset Z80 & YM2612 */492z80_reset();493fm_reset(cycles);494}495496/* update Z80 bus status */497zstate |= 1;498}499else /* !ZRESET asserted */500{501/* check if Z80 is going to be stopped */502if (zstate == 1)503{504/* resynchronize with 68k */505z80_run(cycles);506}507508/* check if 68k had access to Z80 bus */509else if (zstate == 3)510{511/* disable 68k access to Z80 bus */512m68k.memory_map[0xa0].read8 = m68k_read_bus_8;513m68k.memory_map[0xa0].read16 = m68k_read_bus_16;514m68k.memory_map[0xa0].write8 = m68k_unused_8_w;515m68k.memory_map[0xa0].write16 = m68k_unused_16_w;516}517518/* stop YM2612 */519fm_reset(cycles);520521/* update Z80 bus status */522zstate &= 2;523}524}525526void gen_zbank_w (unsigned int data)527{528zbank = ((zbank >> 1) | ((data & 1) << 23)) & 0xFF8000;529}530531532/*-----------------------------------------------------------------------*/533/* Z80 interrupt callback */534/* ----------------------------------------------------------------------*/535536int z80_irq_callback (int param)537{538return -1;539}540541542