/*************************************************************************************** * Genesis Plus * Virtual System emulation * * Copyright (C) 1998, 1999, 2000, 2001, 2002, 2003 Charles Mac Donald (original code) * Eke-Eke (2007-2011), additional code & fixes for the GCN/Wii port * * This program is free software; you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation; either version 2 of the License, or * (at your option) any later version. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program; if not, write to the Free Software * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA * ****************************************************************************************/ #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_md(int do_skip); static void system_frame_sms(int do_skip); static int 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(); system_frame = (system_hw == SYSTEM_PBC) ? system_frame_sms : system_frame_md; } /**************************************************************** * Virtual System emulation ****************************************************************/ void system_reset(void) { gen_reset(1); io_reset(); vdp_reset(); render_reset(); sound_reset(); audio_reset(); } void system_shutdown (void) { gen_shutdown(); SN76489_Shutdown(); } static void system_frame_md(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 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 = (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) { 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_irq_state |= 0x14; } } /* 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_irq_state |= 0x14; } } /* 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_irq_state = 0x16; } /* 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; /* 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); } /* Detect pause button input */ 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); /* 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) { 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); } /* latch Horizontal Scroll register (if modified during VBLANK) */ hscroll = reg[0x08]; /* 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 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) { /* IRQ line is latched between instructions, on 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; } } /* update VDP DMA */ if (dma_length) { vdp_dma_update(mcycles_vdp); } /* render scanline */ if (!do_skip) { render_line(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; /* 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) { Z80.irq_state = ASSERT_LINE; } } /* 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(); /* run Z80 until end of line */ z80_run(mcycles_vdp + MCYCLES_PER_LINE); /* 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(); /* 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; }