mirror of
https://github.com/ekeeke/Genesis-Plus-GX.git
synced 2024-12-29 04:31:49 +01:00
62f1204476
Genesis Plus GX 1.6.0 ---------------------- [Core/Sound] --------------- * added YM2413 emulation in Master System compatibility mode. * fixed SN76489 noise boost initialization. * minor YM2612 core optimizations. [Core/VDP] --------------- * added accurate emulation of SG-1000, Master System (315-5124, 315-5246) & Game Gear VDP. * added support for all TMS9918 rendering modes. * improved Mega Drive VDP timings accuracy in Master System Compatibility mode. * fixed color palette initialization. * fixed shifted sprites rendering in Mode 4. * modified pixel rendering support (pixel depth is now forced at compilation time). [Core/CPU] --------------- * optimized 68k core (rewrote 68k interrupt handling, removed multiple CPU types support & unused code) for 5~8% speed improvment [Core/IO] --------------- * added accurate emulation of Master System (315-5216, 315-5237, 315-5297) & Game Gear I/O controllers. * added Terebi Oekaki tablet emulation. * improved Mouse emulation (fixes mouse support in Cannon Fodder). * improved Justifier emulation (fixes gun support in Lethal Enforcers 2). * improved 6-Buttons control pad emulation (fixes Duke Nukem 3D) * modified lightgun emulation to use common key inputs for all devices. * 2-buttons controller is now picked by default for Master System games. [Core/MD] --------------- * added copy-protection hardware emulation for some new dumped games (Tiny Toon Adventures 3, Mighty Morphin Power Rangers & The Battle of Red Cliffs). * added Game Toshokan in EEPROM database (verified on real cartridge). * fixed Micro Machines 2 - Turbo Tournament EEPROM size (verified on real cartridge). * modified SRAM banswitch hardware emulation to be more compatible with some hacks. [Core/MS] --------------- * added Cyborg Z to Korean mapper database. [Core/GG] --------------- * added 93C46 EEPROM emulation (Majors Pro Baseball, World Series Baseball & World Series Baseball 95). [Core/General] --------------- * added support for .mdx ROM format. * added Game Gear & SG-1000 ROM support. * added accurate emulation of SG-1000, Master System (I, II) & Game Gear hardware models for 100% compatibility. * updated to new Genesis Plus license (see http://cgfm2.emuviews.com/) * removed DOS port * various code cleanup. [Gamecube/Wii] --------------- * IMPORTANT: cheats, screenshots & save files are now stored in console-specific directories (ex: /snaps/md, /cheats/ms, /saves/gg, ...) * added 8-bit Action Replay & Game Genie codes support (for Master System & Game Gear games). * improved audio/video synchronization for PAL games in 50Hz TV modes (now use VSYNC like NTSC games in 60hz modes). * improved gun cursor positioning accuracy. * improved horizontal scaling & screenshots rendering in H32 mode. * fixed a bug with ROM file extension handling that would affect cheats, snapshots, sram & savestate files. * removed ARAM/injected ROM support (unused). * removed WPAD_ and PAD_ update from VSYNC callback. * increased GCC inlining limits for some speed improvment. * compiled with devkitPPC r24 & libogc 1.8.7.
1003 lines
23 KiB
C
1003 lines
23 KiB
C
/***************************************************************************************
|
|
* Genesis Plus
|
|
* Virtual System emulation
|
|
*
|
|
* Support for Genesis & Master System compatibility modes
|
|
*
|
|
* Copyright (C) 1998, 1999, 2000, 2001, 2002, 2003 Charles Mac Donald (original code)
|
|
* Copyright (C) 2007-2011 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"
|
|
#include "Fir_Resampler.h"
|
|
#include "eq.h"
|
|
|
|
/* Global variables */
|
|
t_bitmap bitmap;
|
|
t_snd snd;
|
|
uint32 mcycles_vdp;
|
|
uint32 mcycles_z80;
|
|
uint32 mcycles_68k;
|
|
uint8 system_hw;
|
|
void (*system_frame)(int do_skip);
|
|
|
|
static void system_frame_gen(int do_skip);
|
|
static void system_frame_sms(int do_skip);
|
|
static uint8 pause_b;
|
|
static EQSTATE eq;
|
|
static int32 llp,rrp;
|
|
|
|
/****************************************************************
|
|
* Audio subsystem
|
|
****************************************************************/
|
|
|
|
int audio_init (int samplerate, float framerate)
|
|
{
|
|
/* Shutdown first */
|
|
audio_shutdown();
|
|
|
|
/* Clear the sound data context */
|
|
memset(&snd, 0, sizeof (snd));
|
|
|
|
/* Default settings */
|
|
snd.sample_rate = samplerate;
|
|
snd.frame_rate = framerate;
|
|
|
|
/* Calculate the sound buffer size (for one frame) */
|
|
snd.buffer_size = (int)(samplerate / framerate) + 32;
|
|
|
|
/* SN76489 stream buffers */
|
|
snd.psg.buffer = (int16 *) malloc(snd.buffer_size * sizeof(int16));
|
|
if (!snd.psg.buffer) return (-1);
|
|
|
|
/* YM2612 stream buffers */
|
|
snd.fm.buffer = (int32 *) malloc(snd.buffer_size * sizeof(int32) * 2);
|
|
if (!snd.fm.buffer) return (-1);
|
|
|
|
#ifndef NGC
|
|
/* Output buffers */
|
|
snd.buffer[0] = (int16 *) malloc(snd.buffer_size * sizeof(int16));
|
|
snd.buffer[1] = (int16 *) malloc(snd.buffer_size * sizeof(int16));
|
|
if (!snd.buffer[0] || !snd.buffer[1]) return (-1);
|
|
#endif
|
|
|
|
/* Resampling buffer */
|
|
if (config.hq_fm && !Fir_Resampler_initialize(4096)) return (-1);
|
|
|
|
/* Set audio enable flag */
|
|
snd.enabled = 1;
|
|
|
|
/* Reset audio */
|
|
audio_reset();
|
|
|
|
return (0);
|
|
}
|
|
|
|
void audio_reset(void)
|
|
{
|
|
/* Low-Pass filter */
|
|
llp = 0;
|
|
rrp = 0;
|
|
|
|
/* 3 band EQ */
|
|
audio_set_equalizer();
|
|
|
|
/* Resampling buffer */
|
|
Fir_Resampler_clear();
|
|
|
|
/* Audio buffers */
|
|
snd.psg.pos = snd.psg.buffer;
|
|
snd.fm.pos = snd.fm.buffer;
|
|
if (snd.psg.buffer) memset (snd.psg.buffer, 0, snd.buffer_size * sizeof(int16));
|
|
if (snd.fm.buffer) memset (snd.fm.buffer, 0, snd.buffer_size * sizeof(int32) * 2);
|
|
#ifndef NGC
|
|
if (snd.buffer[0]) memset (snd.buffer[0], 0, snd.buffer_size * sizeof(int16));
|
|
if (snd.buffer[1]) memset (snd.buffer[1], 0, snd.buffer_size * sizeof(int16));
|
|
#endif
|
|
}
|
|
|
|
void audio_set_equalizer(void)
|
|
{
|
|
init_3band_state(&eq,config.low_freq,config.high_freq,snd.sample_rate);
|
|
eq.lg = (double)(config.lg) / 100.0;
|
|
eq.mg = (double)(config.mg) / 100.0;
|
|
eq.hg = (double)(config.hg) / 100.0;
|
|
}
|
|
|
|
void audio_shutdown(void)
|
|
{
|
|
/* Sound buffers */
|
|
if (snd.fm.buffer) free(snd.fm.buffer);
|
|
if (snd.psg.buffer) free(snd.psg.buffer);
|
|
#ifndef NGC
|
|
if (snd.buffer[0]) free(snd.buffer[0]);
|
|
if (snd.buffer[1]) free(snd.buffer[1]);
|
|
#endif
|
|
|
|
/* Resampling buffer */
|
|
Fir_Resampler_shutdown();
|
|
}
|
|
|
|
int audio_update (void)
|
|
{
|
|
int32 i, l, r;
|
|
int32 ll = llp;
|
|
int32 rr = rrp;
|
|
|
|
int psg_preamp = config.psg_preamp;
|
|
int fm_preamp = config.fm_preamp;
|
|
int filter = config.filter;
|
|
uint32 factora = (config.lp_range << 16) / 100;
|
|
uint32 factorb = 0x10000 - factora;
|
|
|
|
int32 *fm = snd.fm.buffer;
|
|
int16 *psg = snd.psg.buffer;
|
|
|
|
#ifdef NGC
|
|
int16 *sb = (int16 *) soundbuffer[mixbuffer];
|
|
#endif
|
|
|
|
/* get number of available samples */
|
|
int size = sound_update(mcycles_vdp);
|
|
|
|
/* return an aligned number of samples */
|
|
size &= ~7;
|
|
|
|
if (config.hq_fm)
|
|
{
|
|
/* resample into FM output buffer */
|
|
Fir_Resampler_read(fm, size);
|
|
|
|
#ifdef LOGSOUND
|
|
error("%d FM samples remaining\n",Fir_Resampler_written() >> 1);
|
|
#endif
|
|
}
|
|
else
|
|
{
|
|
/* adjust remaining samples in FM output buffer*/
|
|
snd.fm.pos -= (size * 2);
|
|
|
|
#ifdef LOGSOUND
|
|
error("%d FM samples remaining\n",(snd.fm.pos - snd.fm.buffer)>>1);
|
|
#endif
|
|
}
|
|
|
|
/* adjust remaining samples in PSG output buffer*/
|
|
snd.psg.pos -= size;
|
|
|
|
#ifdef LOGSOUND
|
|
error("%d PSG samples remaining\n",snd.psg.pos - snd.psg.buffer);
|
|
#endif
|
|
|
|
/* mix samples */
|
|
for (i = 0; i < size; i ++)
|
|
{
|
|
/* PSG samples (mono) */
|
|
l = r = (((*psg++) * psg_preamp) / 100);
|
|
|
|
/* FM samples (stereo) */
|
|
l += ((*fm++ * fm_preamp) / 100);
|
|
r += ((*fm++ * fm_preamp) / 100);
|
|
|
|
/* filtering */
|
|
if (filter & 1)
|
|
{
|
|
/* single-pole low-pass filter (6 dB/octave) */
|
|
ll = (ll>>16)*factora + l*factorb;
|
|
rr = (rr>>16)*factora + r*factorb;
|
|
l = ll >> 16;
|
|
r = rr >> 16;
|
|
}
|
|
else if (filter & 2)
|
|
{
|
|
/* 3 Band EQ */
|
|
l = do_3band(&eq,l);
|
|
r = do_3band(&eq,r);
|
|
}
|
|
|
|
/* clipping (16-bit samples) */
|
|
if (l > 32767) l = 32767;
|
|
else if (l < -32768) l = -32768;
|
|
if (r > 32767) r = 32767;
|
|
else if (r < -32768) r = -32768;
|
|
|
|
/* update sound buffer */
|
|
#ifndef NGC
|
|
snd.buffer[0][i] = r;
|
|
snd.buffer[1][i] = l;
|
|
#else
|
|
*sb++ = r;
|
|
*sb++ = l;
|
|
#endif
|
|
}
|
|
|
|
/* save filtered samples for next frame */
|
|
llp = ll;
|
|
rrp = rr;
|
|
|
|
/* keep remaining samples for next frame */
|
|
memcpy(snd.fm.buffer, fm, (snd.fm.pos - snd.fm.buffer) * 4);
|
|
memcpy(snd.psg.buffer, psg, (snd.psg.pos - snd.psg.buffer) * 2);
|
|
|
|
#ifdef LOGSOUND
|
|
error("%d samples returned\n\n",size);
|
|
#endif
|
|
|
|
return size;
|
|
}
|
|
|
|
/****************************************************************
|
|
* Virtual Genesis initialization
|
|
****************************************************************/
|
|
void system_init(void)
|
|
{
|
|
gen_init();
|
|
io_init();
|
|
vdp_init();
|
|
render_init();
|
|
sound_init();
|
|
|
|
switch (system_hw)
|
|
{
|
|
case SYSTEM_MD:
|
|
case SYSTEM_PICO:
|
|
{
|
|
system_frame = system_frame_gen;
|
|
break;
|
|
}
|
|
|
|
default:
|
|
{
|
|
system_frame = system_frame_sms;
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
/****************************************************************
|
|
* Virtual System emulation
|
|
****************************************************************/
|
|
void system_reset(void)
|
|
{
|
|
gen_reset(1);
|
|
io_reset();
|
|
render_reset();
|
|
vdp_reset();
|
|
sound_reset();
|
|
audio_reset();
|
|
}
|
|
|
|
void system_shutdown (void)
|
|
{
|
|
gen_shutdown();
|
|
SN76489_Shutdown();
|
|
}
|
|
|
|
static void system_frame_gen(int do_skip)
|
|
{
|
|
/* line counter */
|
|
int line = 0;
|
|
|
|
/* Z80 interrupt flag */
|
|
int zirq = 1;
|
|
|
|
/* reload H Counter */
|
|
int h_counter = reg[10];
|
|
|
|
/* reset line master cycle count */
|
|
mcycles_vdp = 0;
|
|
|
|
/* reload V Counter */
|
|
v_counter = lines_per_frame - 1;
|
|
|
|
/* reset VDP FIFO */
|
|
fifo_write_cnt = 0;
|
|
fifo_lastwrite = 0;
|
|
|
|
/* update 6-Buttons & Lightguns */
|
|
input_refresh();
|
|
|
|
/* display changed during VBLANK */
|
|
if (bitmap.viewport.changed & 2)
|
|
{
|
|
bitmap.viewport.changed &= ~2;
|
|
|
|
/* interlaced modes */
|
|
int old_interlaced = interlaced;
|
|
interlaced = (reg[12] & 0x02) >> 1;
|
|
if (old_interlaced != interlaced)
|
|
{
|
|
/* double resolution mode */
|
|
im2_flag = ((reg[12] & 0x06) == 0x06);
|
|
|
|
/* reset field status flag */
|
|
odd_frame = 1;
|
|
|
|
/* video mode has changed */
|
|
bitmap.viewport.changed = 5;
|
|
|
|
/* update rendering mode */
|
|
if (reg[1] & 0x04)
|
|
{
|
|
if (im2_flag)
|
|
{
|
|
render_bg = (reg[11] & 0x04) ? render_bg_m5_im2_vs : render_bg_m5_im2;
|
|
render_obj = (reg[12] & 0x08) ? render_obj_m5_im2_ste : render_obj_m5_im2;
|
|
}
|
|
else
|
|
{
|
|
render_bg = (reg[11] & 0x04) ? render_bg_m5_vs : render_bg_m5;
|
|
render_obj = (reg[12] & 0x08) ? render_obj_m5_ste : render_obj_m5;
|
|
}
|
|
}
|
|
}
|
|
|
|
/* active screen height */
|
|
if (reg[1] & 0x04)
|
|
{
|
|
bitmap.viewport.h = 224 + ((reg[1] & 0x08) << 1);
|
|
bitmap.viewport.y = (config.overscan & 1) * ((240 + 48*vdp_pal - bitmap.viewport.h) >> 1);
|
|
}
|
|
else
|
|
{
|
|
bitmap.viewport.h = 192;
|
|
bitmap.viewport.y = (config.overscan & 1) * 24 * (vdp_pal + 1);
|
|
}
|
|
|
|
/* active screen width */
|
|
bitmap.viewport.w = 256 + ((reg[12] & 0x01) << 6);
|
|
}
|
|
|
|
/* clear VBLANK, DMA, FIFO FULL & field flags */
|
|
status &= 0xFEE5;
|
|
|
|
/* set FIFO EMPTY flag */
|
|
status |= 0x0200;
|
|
|
|
/* even/odd field flag (interlaced modes only) */
|
|
odd_frame ^= 1;
|
|
if (interlaced)
|
|
{
|
|
status |= (odd_frame << 4);
|
|
}
|
|
|
|
/* update VDP DMA */
|
|
if (dma_length)
|
|
{
|
|
vdp_dma_update(0);
|
|
}
|
|
|
|
/* render last line of overscan */
|
|
if (bitmap.viewport.y > 0)
|
|
{
|
|
blank_line(v_counter, -bitmap.viewport.x, bitmap.viewport.w + 2*bitmap.viewport.x);
|
|
}
|
|
|
|
/* parse first line of sprites */
|
|
if (reg[1] & 0x40)
|
|
{
|
|
parse_satb(-1);
|
|
}
|
|
|
|
/* run 68k & Z80 */
|
|
m68k_run(MCYCLES_PER_LINE);
|
|
if (zstate == 1)
|
|
{
|
|
z80_run(MCYCLES_PER_LINE);
|
|
}
|
|
else
|
|
{
|
|
mcycles_z80 = MCYCLES_PER_LINE;
|
|
}
|
|
|
|
/* run SVP chip */
|
|
if (svp)
|
|
{
|
|
ssp1601_run(SVP_cycles);
|
|
}
|
|
|
|
/* update line cycle count */
|
|
mcycles_vdp += MCYCLES_PER_LINE;
|
|
|
|
/* Active Display */
|
|
do
|
|
{
|
|
/* update V Counter */
|
|
v_counter = line;
|
|
|
|
/* update 6-Buttons & Lightguns */
|
|
input_refresh();
|
|
|
|
/* H Interrupt */
|
|
if(--h_counter < 0)
|
|
{
|
|
/* reload H Counter */
|
|
h_counter = reg[10];
|
|
|
|
/* interrupt level 4 */
|
|
hint_pending = 0x10;
|
|
if (reg[0] & 0x10)
|
|
{
|
|
m68k_update_irq(4);
|
|
}
|
|
}
|
|
|
|
/* update VDP DMA */
|
|
if (dma_length)
|
|
{
|
|
vdp_dma_update(mcycles_vdp);
|
|
}
|
|
|
|
/* render scanline */
|
|
if (!do_skip)
|
|
{
|
|
render_line(line);
|
|
}
|
|
|
|
/* run 68k & Z80 */
|
|
m68k_run(mcycles_vdp + MCYCLES_PER_LINE);
|
|
if (zstate == 1)
|
|
{
|
|
z80_run(mcycles_vdp + MCYCLES_PER_LINE);
|
|
}
|
|
else
|
|
{
|
|
mcycles_z80 = mcycles_vdp + MCYCLES_PER_LINE;
|
|
}
|
|
|
|
/* run SVP chip */
|
|
if (svp)
|
|
{
|
|
ssp1601_run(SVP_cycles);
|
|
}
|
|
|
|
/* update line cycle count */
|
|
mcycles_vdp += MCYCLES_PER_LINE;
|
|
}
|
|
while (++line < bitmap.viewport.h);
|
|
|
|
/* end of active display */
|
|
v_counter = line;
|
|
|
|
/* set VBLANK flag */
|
|
status |= 0x08;
|
|
|
|
/* overscan area */
|
|
int start = lines_per_frame - bitmap.viewport.y;
|
|
int end = bitmap.viewport.h + bitmap.viewport.y;
|
|
|
|
/* check viewport changes */
|
|
if ((bitmap.viewport.w != bitmap.viewport.ow) || (bitmap.viewport.h != bitmap.viewport.oh))
|
|
{
|
|
bitmap.viewport.ow = bitmap.viewport.w;
|
|
bitmap.viewport.oh = bitmap.viewport.h;
|
|
bitmap.viewport.changed |= 1;
|
|
}
|
|
|
|
/* update 6-Buttons & Lightguns */
|
|
input_refresh();
|
|
|
|
/* H Interrupt */
|
|
if(--h_counter < 0)
|
|
{
|
|
/* reload H Counter */
|
|
h_counter = reg[10];
|
|
|
|
/* interrupt level 4 */
|
|
hint_pending = 0x10;
|
|
if (reg[0] & 0x10)
|
|
{
|
|
m68k_update_irq(4);
|
|
}
|
|
}
|
|
|
|
/* update VDP DMA */
|
|
if (dma_length)
|
|
{
|
|
vdp_dma_update(mcycles_vdp);
|
|
}
|
|
|
|
/* render overscan */
|
|
if (line < end)
|
|
{
|
|
blank_line(line, -bitmap.viewport.x, bitmap.viewport.w + 2*bitmap.viewport.x);
|
|
}
|
|
|
|
/* update inputs before VINT (Warriors of Eternal Sun) */
|
|
osd_input_Update();
|
|
|
|
/* delay between VINT flag & V Interrupt (Ex-Mutants, Tyrant) */
|
|
m68k_run(mcycles_vdp + 588);
|
|
status |= 0x80;
|
|
|
|
/* delay between VBLANK flag & V Interrupt (Dracula, OutRunners, VR Troopers) */
|
|
m68k_run(mcycles_vdp + 788);
|
|
if (zstate == 1)
|
|
{
|
|
z80_run(mcycles_vdp + 788);
|
|
}
|
|
else
|
|
{
|
|
mcycles_z80 = mcycles_vdp + 788;
|
|
}
|
|
|
|
/* V Interrupt */
|
|
vint_pending = 0x20;
|
|
if (reg[1] & 0x20)
|
|
{
|
|
m68k_set_irq(6);
|
|
}
|
|
|
|
/* assert Z80 interrupt */
|
|
Z80.irq_state = ASSERT_LINE;
|
|
|
|
/* run 68k & Z80 until end of line */
|
|
m68k_run(mcycles_vdp + MCYCLES_PER_LINE);
|
|
if (zstate == 1)
|
|
{
|
|
z80_run(mcycles_vdp + MCYCLES_PER_LINE);
|
|
}
|
|
else
|
|
{
|
|
mcycles_z80 = mcycles_vdp + MCYCLES_PER_LINE;
|
|
}
|
|
|
|
/* run SVP chip */
|
|
if (svp)
|
|
{
|
|
ssp1601_run(SVP_cycles);
|
|
}
|
|
|
|
/* update line cycle count */
|
|
mcycles_vdp += MCYCLES_PER_LINE;
|
|
|
|
/* increment line count */
|
|
line++;
|
|
|
|
/* Vertical Blanking */
|
|
do
|
|
{
|
|
/* update V Counter */
|
|
v_counter = line;
|
|
|
|
/* update 6-Buttons & Lightguns */
|
|
input_refresh();
|
|
|
|
/* render overscan */
|
|
if ((line < end) || (line >= start))
|
|
{
|
|
blank_line(line, -bitmap.viewport.x, bitmap.viewport.w + 2*bitmap.viewport.x);
|
|
}
|
|
|
|
if (zirq)
|
|
{
|
|
/* Z80 interrupt is asserted exactly for one line */
|
|
m68k_run(mcycles_vdp + 788);
|
|
if (zstate == 1)
|
|
{
|
|
z80_run(mcycles_vdp + 788);
|
|
}
|
|
else
|
|
{
|
|
mcycles_z80 = mcycles_vdp + 788;
|
|
}
|
|
|
|
/* clear Z80 interrupt */
|
|
Z80.irq_state = CLEAR_LINE;
|
|
zirq = 0;
|
|
}
|
|
|
|
/* run 68k & Z80 */
|
|
m68k_run(mcycles_vdp + MCYCLES_PER_LINE);
|
|
if (zstate == 1)
|
|
{
|
|
z80_run(mcycles_vdp + MCYCLES_PER_LINE);
|
|
}
|
|
else
|
|
{
|
|
mcycles_z80 = mcycles_vdp + MCYCLES_PER_LINE;
|
|
}
|
|
|
|
/* run SVP chip */
|
|
if (svp)
|
|
{
|
|
ssp1601_run(SVP_cycles);
|
|
}
|
|
|
|
/* update line cycle count */
|
|
mcycles_vdp += MCYCLES_PER_LINE;
|
|
}
|
|
while (++line < (lines_per_frame - 1));
|
|
|
|
/* adjust 68k & Z80 cycle count for next frame */
|
|
mcycles_68k -= mcycles_vdp;
|
|
mcycles_z80 -= mcycles_vdp;
|
|
}
|
|
|
|
|
|
static void system_frame_sms(int do_skip)
|
|
{
|
|
/* line counter */
|
|
int line = 0;
|
|
|
|
/* reload H Counter */
|
|
int h_counter = reg[10];
|
|
|
|
/* reset line master cycle count */
|
|
mcycles_vdp = 0;
|
|
|
|
/* reload V Counter */
|
|
v_counter = lines_per_frame - 1;
|
|
|
|
/* reset VDP FIFO */
|
|
fifo_write_cnt = 0;
|
|
fifo_lastwrite = 0;
|
|
|
|
/* update 6-Buttons & Lightguns */
|
|
input_refresh();
|
|
|
|
/* display changed during VBLANK */
|
|
if (bitmap.viewport.changed & 2)
|
|
{
|
|
bitmap.viewport.changed &= ~2;
|
|
|
|
if (system_hw & SYSTEM_MD)
|
|
{
|
|
/* interlaced mode */
|
|
int old_interlaced = interlaced;
|
|
interlaced = (reg[12] & 0x02) >> 1;
|
|
if (old_interlaced != interlaced)
|
|
{
|
|
im2_flag = ((reg[12] & 0x06) == 0x06);
|
|
odd_frame = 1;
|
|
bitmap.viewport.changed = 5;
|
|
|
|
/* update rendering mode */
|
|
if (reg[1] & 0x04)
|
|
{
|
|
if (im2_flag)
|
|
{
|
|
render_bg = (reg[11] & 0x04) ? render_bg_m5_im2_vs : render_bg_m5_im2;
|
|
render_obj = render_obj_m5_im2;
|
|
|
|
}
|
|
else
|
|
{
|
|
render_bg = (reg[11] & 0x04) ? render_bg_m5_vs : render_bg_m5;
|
|
render_obj = render_obj_m5;
|
|
}
|
|
}
|
|
}
|
|
|
|
/* active screen height */
|
|
if (reg[1] & 0x04)
|
|
{
|
|
bitmap.viewport.h = 224 + ((reg[1] & 0x08) << 1);
|
|
bitmap.viewport.y = (config.overscan & 1) * ((240 + 48*vdp_pal - bitmap.viewport.h) >> 1);
|
|
}
|
|
else
|
|
{
|
|
bitmap.viewport.h = 192;
|
|
bitmap.viewport.y = (config.overscan & 1) * 24 * (vdp_pal + 1);
|
|
}
|
|
|
|
/* active screen width */
|
|
bitmap.viewport.w = 256 + ((reg[12] & 0x01) << 6);
|
|
}
|
|
else
|
|
{
|
|
/* check for VDP extended modes */
|
|
int mode = (reg[0] & 0x06) | (reg[1] & 0x18);
|
|
|
|
/* update active height */
|
|
if (mode == 0x0E)
|
|
{
|
|
bitmap.viewport.h = 240;
|
|
}
|
|
else if (mode == 0x16)
|
|
{
|
|
bitmap.viewport.h = 224;
|
|
}
|
|
else
|
|
{
|
|
bitmap.viewport.h = 192;
|
|
}
|
|
|
|
/* update vertical overscan */
|
|
if (config.overscan & 1)
|
|
{
|
|
bitmap.viewport.y = (240 + 48*vdp_pal - bitmap.viewport.h) >> 1;
|
|
}
|
|
else
|
|
{
|
|
if (system_hw == SYSTEM_GG)
|
|
{
|
|
/* Display area reduced to 160x144 */
|
|
bitmap.viewport.y = (144 - bitmap.viewport.h) / 2;
|
|
}
|
|
else
|
|
{
|
|
bitmap.viewport.y = 0;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
/* Detect pause button input (in Game Gear Mode, NMI is not generated) */
|
|
if ((system_hw != SYSTEM_GG) && (system_hw != SYSTEM_SG))
|
|
{
|
|
if (input.pad[0] & INPUT_START)
|
|
{
|
|
/* NMI is edge-triggered */
|
|
if (!pause_b)
|
|
{
|
|
pause_b = 1;
|
|
z80_set_nmi_line(ASSERT_LINE);
|
|
z80_set_nmi_line(CLEAR_LINE);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
pause_b = 0;
|
|
}
|
|
}
|
|
|
|
/* 3-D glasses faking: skip rendering of left lens frame */
|
|
do_skip |= (work_ram[0x1ffb] & cart.special);
|
|
|
|
/* Mega Drive VDP specific */
|
|
if (system_hw & SYSTEM_MD)
|
|
{
|
|
/* clear VBLANK, DMA & field flags */
|
|
status &= 0xE5;
|
|
|
|
/* even/odd field flag (interlaced modes only) */
|
|
odd_frame ^= 1;
|
|
if (interlaced)
|
|
{
|
|
status |= (odd_frame << 4);
|
|
}
|
|
|
|
/* update VDP DMA */
|
|
if (dma_length)
|
|
{
|
|
vdp_dma_update(0);
|
|
}
|
|
}
|
|
|
|
/* Master System & Game Gear VDP specific */
|
|
if (system_hw < SYSTEM_MD)
|
|
{
|
|
/* Sprites are still processed during vertical borders */
|
|
if (reg[1] & 0x40)
|
|
{
|
|
render_obj(bitmap.viewport.w);
|
|
}
|
|
}
|
|
|
|
/* render last line of overscan */
|
|
if (bitmap.viewport.y > 0)
|
|
{
|
|
blank_line(v_counter, -bitmap.viewport.x, bitmap.viewport.w + 2*bitmap.viewport.x);
|
|
}
|
|
|
|
/* parse first line of sprites (on Master System VDP, pre-processing still occurs when display is disabled) */
|
|
if ((reg[1] & 0x40) || (system_hw < SYSTEM_MD))
|
|
{
|
|
parse_satb(-1);
|
|
}
|
|
|
|
/* run Z80 */
|
|
z80_run(MCYCLES_PER_LINE);
|
|
|
|
/* update line cycle count */
|
|
mcycles_vdp += MCYCLES_PER_LINE;
|
|
|
|
/* latch Vertical Scroll register */
|
|
vscroll = reg[0x09];
|
|
|
|
/* Active Display */
|
|
do
|
|
{
|
|
/* update VDP DMA (Mega Drive VDP specific) */
|
|
if (dma_length)
|
|
{
|
|
vdp_dma_update(mcycles_vdp);
|
|
}
|
|
|
|
/* make sure we didn't already render that line */
|
|
if (v_counter != line)
|
|
{
|
|
/* update V Counter */
|
|
v_counter = line;
|
|
|
|
/* render scanline */
|
|
if (!do_skip)
|
|
{
|
|
render_line(line);
|
|
}
|
|
}
|
|
|
|
/* update 6-Buttons & Lightguns */
|
|
input_refresh();
|
|
|
|
/* H Interrupt */
|
|
if(--h_counter < 0)
|
|
{
|
|
/* reload H Counter */
|
|
h_counter = reg[10];
|
|
|
|
/* interrupt level 4 */
|
|
hint_pending = 0x10;
|
|
if (reg[0] & 0x10)
|
|
{
|
|
/* cycle-accurate HINT */
|
|
/* IRQ line is latched between instructions, during instruction last cycle. */
|
|
/* This means that if Z80 cycle count is exactly a multiple of MCYCLES_PER_LINE, */
|
|
/* interrupt should be triggered AFTER the next instruction. */
|
|
if ((mcycles_z80 % MCYCLES_PER_LINE) == 0)
|
|
{
|
|
z80_run(mcycles_z80 + 1);
|
|
}
|
|
|
|
Z80.irq_state = ASSERT_LINE;
|
|
}
|
|
}
|
|
|
|
/* run Z80 */
|
|
z80_run(mcycles_vdp + MCYCLES_PER_LINE);
|
|
|
|
/* update line cycle count */
|
|
mcycles_vdp += MCYCLES_PER_LINE;
|
|
}
|
|
while (++line < bitmap.viewport.h);
|
|
|
|
/* end of active display */
|
|
v_counter = line;
|
|
|
|
/* Mega Drive VDP specific */
|
|
if (system_hw & SYSTEM_MD)
|
|
{
|
|
/* set VBLANK flag */
|
|
status |= 0x08;
|
|
}
|
|
|
|
/* overscan area */
|
|
int start = lines_per_frame - bitmap.viewport.y;
|
|
int end = bitmap.viewport.h + bitmap.viewport.y;
|
|
|
|
/* check viewport changes */
|
|
if ((bitmap.viewport.w != bitmap.viewport.ow) || (bitmap.viewport.h != bitmap.viewport.oh))
|
|
{
|
|
bitmap.viewport.ow = bitmap.viewport.w;
|
|
bitmap.viewport.oh = bitmap.viewport.h;
|
|
bitmap.viewport.changed |= 1;
|
|
}
|
|
|
|
/* update 6-Buttons & Lightguns */
|
|
input_refresh();
|
|
|
|
/* H Interrupt */
|
|
if(--h_counter < 0)
|
|
{
|
|
/* reload H Counter */
|
|
h_counter = reg[10];
|
|
|
|
/* interrupt level 4 */
|
|
hint_pending = 0x10;
|
|
if (reg[0] & 0x10)
|
|
{
|
|
/* cycle-accurate HINT */
|
|
if ((mcycles_z80 % MCYCLES_PER_LINE) == 0)
|
|
{
|
|
z80_run(mcycles_z80 + 1);
|
|
}
|
|
|
|
Z80.irq_state = ASSERT_LINE;
|
|
}
|
|
}
|
|
|
|
/* update VDP DMA (Mega Drive VDP specific) */
|
|
if (dma_length)
|
|
{
|
|
vdp_dma_update(mcycles_vdp);
|
|
}
|
|
|
|
/* render overscan */
|
|
if (line < end)
|
|
{
|
|
blank_line(line, -bitmap.viewport.x, bitmap.viewport.w + 2*bitmap.viewport.x);
|
|
}
|
|
|
|
/* update inputs before VINT */
|
|
osd_input_Update();
|
|
|
|
/* run Z80 until end of line */
|
|
z80_run(mcycles_vdp + MCYCLES_PER_LINE);
|
|
|
|
/* make sure VINT flag was not cleared by last instruction */
|
|
if (v_counter == line)
|
|
{
|
|
/* Set VINT flag */
|
|
status |= 0x80;
|
|
|
|
/* V Interrupt */
|
|
vint_pending = 0x20;
|
|
if (reg[1] & 0x20)
|
|
{
|
|
Z80.irq_state = ASSERT_LINE;
|
|
}
|
|
}
|
|
|
|
/* update line cycle count */
|
|
mcycles_vdp += MCYCLES_PER_LINE;
|
|
|
|
/* increment line count */
|
|
line++;
|
|
|
|
/* Vertical Blanking */
|
|
do
|
|
{
|
|
/* update V Counter */
|
|
v_counter = line;
|
|
|
|
/* update 6-Buttons & Lightguns */
|
|
input_refresh();
|
|
|
|
/* Master System & Game Gear VDP specific */
|
|
if ((system_hw < SYSTEM_MD) && (line > (lines_per_frame - 16)))
|
|
{
|
|
/* Sprites are still processed during top border */
|
|
render_obj(bitmap.viewport.w);
|
|
parse_satb(line - lines_per_frame);
|
|
}
|
|
|
|
/* render overscan */
|
|
if ((line < end) || (line >= start))
|
|
{
|
|
blank_line(line, -bitmap.viewport.x, bitmap.viewport.w + 2*bitmap.viewport.x);
|
|
}
|
|
|
|
/* run Z80 */
|
|
z80_run(mcycles_vdp + MCYCLES_PER_LINE);
|
|
|
|
/* update line cycle count */
|
|
mcycles_vdp += MCYCLES_PER_LINE;
|
|
}
|
|
while (++line < (lines_per_frame - 1));
|
|
|
|
/* adjust Z80 cycle count for next frame */
|
|
mcycles_z80 -= mcycles_vdp;
|
|
}
|