From 588881c4f8b8bbd6e91122dd660bcfacbd3a6296 Mon Sep 17 00:00:00 2001 From: kub Date: Fri, 28 Mar 2025 09:27:28 +0100 Subject: [PATCH] mcd, improve cdda save state --- pico/cd/mcd.c | 22 ++++++++++++++++++++-- pico/cd/megasd.c | 1 + pico/pico_int.h | 8 +++++--- pico/sound/sound.c | 30 +++++++++++++++++++++--------- 4 files changed, 47 insertions(+), 14 deletions(-) diff --git a/pico/cd/mcd.c b/pico/cd/mcd.c index 42a5a515..9476f501 100644 --- a/pico/cd/mcd.c +++ b/pico/cd/mcd.c @@ -151,10 +151,10 @@ static void SekRunS68k(unsigned int to) void PicoMCDPrepare(void) { - // ~1.63 for NTSC, ~1.645 for PAL + // 12500000/(osc/7), ~1.63 for NTSC, ~1.645 for PAL #define DIV_ROUND(x,y) ((x)+(y)/2) / (y) // round to nearest, x/y+0.5 -> (x+y/2)/y unsigned int osc = (Pico.m.pal ? OSC_PAL : OSC_NTSC); - mcd_m68k_cycle_mult = DIV_ROUND(12500000ull << 16, osc / 7); + mcd_m68k_cycle_mult = DIV_ROUND(7 * 12500000ull << 16, osc); mcd_s68k_cycle_mult = DIV_ROUND(1ull * osc << 16, 7 * 12500000); } @@ -166,9 +166,27 @@ unsigned int pcd_cycles_m68k_to_s68k(unsigned int c) /* events */ static void pcd_cdc_event(unsigned int now) { + int audio = Pico_mcd->s68k_regs[0x36] & 0x1; + // 75Hz CDC update cdd_update(); + // main 68k cycles since frame start + int cycles = 1LL*(now-mcd_s68k_cycle_base) * mcd_s68k_cycle_mult >> 16; + // samples@rate since frame start + int samples = 1LL * cycles_68k_to_z80(cycles) * Pico.snd.clkz_mult >> 20; + // samples@44100Hz since frame start + samples = samples * Pico.snd.cdda_mult >> 16; + if (samples < 2352/4) // save offset to 1st used sample for state saving + Pico_mcd->m.cdda_lba_offset = 2352/4 - samples; + + /* if audio just turned on, store start offset for sound */ + audio &= !(Pico_mcd->s68k_regs[0x36] & 0x1); + if (audio) { + Pico_mcd->m.cdda_lba_offset = 0; // starting with full lba + Pico_mcd->cdda_frame_offs = samples; + } + /* check if a new CDD command has been processed */ if (!(Pico_mcd->s68k_regs[0x4b] & 0xf0)) { diff --git a/pico/cd/megasd.c b/pico/cd/megasd.c index bc03d17c..4a8f42c3 100644 --- a/pico/cd/megasd.c +++ b/pico/cd/megasd.c @@ -46,6 +46,7 @@ static void cdd_play(s32 lba) { Pico_msd.currentlba = lba; + Pico_mcd->m.cdda_lba_offset = 0; cdd_play_audio(Pico_msd.index, Pico_msd.currentlba); Pico_msd.state |= MSD_ST_PLAY; Pico_msd.state &= ~MSD_ST_PAUSE; diff --git a/pico/pico_int.h b/pico/pico_int.h index c98a02a6..e2fd39ab 100644 --- a/pico/pico_int.h +++ b/pico/pico_int.h @@ -270,7 +270,7 @@ extern SH2 sh2s[2]; // --------------------------------------------------------- // main oscillator clock which controls timing -#define OSC_NTSC 53693100 +#define OSC_NTSC 53693175 #define OSC_PAL 53203424 // PicoVideo.debug_p @@ -475,7 +475,7 @@ struct PicoSound short len_use; // adjusted int len_e_add; // for non-int samples/frame int len_e_cnt; - unsigned int clkl_mult; // z80 clocks per line in Q20 + unsigned int clkz_mult; // z80 clocks per sample in Q20 unsigned int smpl_mult; // samples per line in Q16 unsigned int cdda_mult, cdda_div; // 44.1 KHz resampling factor in Q16 short dac_val, dac_val2; // last DAC sample @@ -547,7 +547,8 @@ struct mcd_misc unsigned char need_sync; unsigned char pad3; unsigned int m68k_poll_clk; - int pad4[8]; + unsigned int cdda_lba_offset; // 20 + int pad4[7]; }; typedef struct @@ -577,6 +578,7 @@ typedef struct struct mcd_pcm pcm; // 112240: void *cdda_stream; int cdda_type; + unsigned int cdda_frame_offs; int pcm_mixbuf[PCM_MIXBUF_LEN * 2]; int pcm_mixpos; char pcm_mixbuf_dirty; diff --git a/pico/sound/sound.c b/pico/sound/sound.c index 26a182f8..455aac7b 100644 --- a/pico/sound/sound.c +++ b/pico/sound/sound.c @@ -189,8 +189,8 @@ void PsndRerate(int preserve_state) // samples per line (Q16) Pico.snd.smpl_mult = 65536LL * PicoIn.sndRate / (target_fps*target_lines); // samples per z80 clock (Q20) - Pico.snd.clkl_mult = 16 * Pico.snd.smpl_mult * 15/7 / 488.5; - // samples per 44.1 KHz sample + Pico.snd.clkz_mult = 16 * Pico.snd.smpl_mult * 15/7 / 488.5; + // samples per 44.1 KHz sample (Q16) Pico.snd.cdda_mult = 65536LL * 44100 / PicoIn.sndRate; Pico.snd.cdda_div = 65536LL * PicoIn.sndRate / 44100; @@ -229,7 +229,7 @@ PICO_INTERNAL void PsndDoDAC(int cyc_to) if (!PicoIn.sndOut) return; // number of samples to fill in buffer (Q20) - len = (cyc_to * Pico.snd.clkl_mult) - Pico.snd.dac_pos; + len = (cyc_to * Pico.snd.clkz_mult) - Pico.snd.dac_pos; // update position and calculate buffer offset and length pos = (Pico.snd.dac_pos+0x80000) >> 20; @@ -271,7 +271,7 @@ PICO_INTERNAL void PsndDoPSG(int cyc_to) if (!PicoIn.sndOut) return; // number of samples to fill in buffer (Q20) - len = (cyc_to * Pico.snd.clkl_mult) - Pico.snd.psg_pos; + len = (cyc_to * Pico.snd.clkz_mult) - Pico.snd.psg_pos; // update position and calculate buffer offset and length pos = (Pico.snd.psg_pos+0x80000) >> 20; @@ -301,7 +301,7 @@ PICO_INTERNAL void PsndDoSMSFM(int cyc_to) if (!PicoIn.sndOut) return; // number of samples to fill in buffer (Q20) - len = (cyc_to * Pico.snd.clkl_mult) - Pico.snd.ym2413_pos; + len = (cyc_to * Pico.snd.clkz_mult) - Pico.snd.ym2413_pos; // update position and calculate buffer offset and length pos = (Pico.snd.ym2413_pos+0x80000) >> 20; @@ -342,7 +342,7 @@ PICO_INTERNAL void PsndDoFM(int cyc_to) if (!PicoIn.sndOut) return; // Q20, number of samples since last call - len = (cyc_to * Pico.snd.clkl_mult) - Pico.snd.fm_pos; + len = (cyc_to * Pico.snd.clkz_mult) - Pico.snd.fm_pos; // update position and calculate buffer offset and length pos = (Pico.snd.fm_pos+0x80000) >> 20; @@ -369,7 +369,7 @@ PICO_INTERNAL void PsndDoPCM(int cyc_to) if (!PicoIn.sndOut) return; // Q20, number of samples since last call - len = (cyc_to * Pico.snd.clkl_mult) - Pico.snd.pcm_pos; + len = (cyc_to * Pico.snd.clkz_mult) - Pico.snd.pcm_pos; // update position and calculate buffer offset and length pos = (Pico.snd.pcm_pos+0x80000) >> 20; @@ -391,13 +391,23 @@ static void cdda_raw_update(s32 *buffer, int length, int stereo) { int ret, cdda_bytes; + // apply start offset in frame (offset to 1st lba to play) + int offs = Pico_mcd->cdda_frame_offs * Pico.snd.cdda_div >> 16; + length -= offs; + buffer += offs * (stereo ? 2 : 1); + Pico_mcd->cdda_frame_offs = 0; + cdda_bytes = (length * Pico.snd.cdda_mult >> 16) * 4; + // compute offset of last played sample in this frame (need for save/load) + Pico_mcd->m.cdda_lba_offset += cdda_bytes/4; + while (Pico_mcd->m.cdda_lba_offset >= 2352/4) + Pico_mcd->m.cdda_lba_offset -= 2352/4; + ret = pm_read_audio(cdda_out_buffer, cdda_bytes, Pico_mcd->cdda_stream); if (ret < cdda_bytes) { memset((char *)cdda_out_buffer + ret, 0, cdda_bytes - ret); Pico_mcd->cdda_stream = NULL; - return; } // now mix @@ -423,7 +433,9 @@ void cdda_start_play(int lba_base, int lba_offset, int lb_len) return; } - pm_seek(Pico_mcd->cdda_stream, (lba_base + lba_offset) * 2352, SEEK_SET); + // on restart after loading, consider offset of last played sample + pm_seek(Pico_mcd->cdda_stream, (lba_base + lba_offset) * 2352 + + Pico_mcd->m.cdda_lba_offset * 4, SEEK_SET); if (Pico_mcd->cdda_type == CT_WAV) { // skip headers, assume it's 44kHz stereo uncompressed -- 2.39.5