mcd, improve cdda save state
authorkub <derkub@gmail.com>
Fri, 28 Mar 2025 08:27:28 +0000 (09:27 +0100)
committerkub <derkub@gmail.com>
Fri, 28 Mar 2025 08:27:28 +0000 (09:27 +0100)
pico/cd/mcd.c
pico/cd/megasd.c
pico/pico_int.h
pico/sound/sound.c

index 42a5a51..9476f50 100644 (file)
@@ -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))
   {
index bc03d17..4a8f42c 100644 (file)
@@ -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;
index c98a02a..e2fd39a 100644 (file)
@@ -270,7 +270,7 @@ extern SH2 sh2s[2];
 // ---------------------------------------------------------\r
 \r
 // main oscillator clock which controls timing\r
-#define OSC_NTSC 53693100\r
+#define OSC_NTSC 53693175\r
 #define OSC_PAL  53203424\r
 \r
 // PicoVideo.debug_p\r
@@ -475,7 +475,7 @@ struct PicoSound
   short len_use;                        // adjusted\r
   int len_e_add;                        // for non-int samples/frame\r
   int len_e_cnt;\r
-  unsigned int clkl_mult;               // z80 clocks per line in Q20\r
+  unsigned int clkz_mult;               // z80 clocks per sample in Q20\r
   unsigned int smpl_mult;               // samples per line in Q16\r
   unsigned int cdda_mult, cdda_div;     // 44.1 KHz resampling factor in Q16\r
   short dac_val, dac_val2;              // last DAC sample\r
@@ -547,7 +547,8 @@ struct mcd_misc
   unsigned char  need_sync;\r
   unsigned char  pad3;\r
   unsigned int   m68k_poll_clk;\r
-  int pad4[8];\r
+  unsigned int   cdda_lba_offset; // 20\r
+  int pad4[7];\r
 };\r
 \r
 typedef struct\r
@@ -577,6 +578,7 @@ typedef struct
   struct mcd_pcm pcm;                          // 112240:\r
   void *cdda_stream;\r
   int cdda_type;\r
+  unsigned int cdda_frame_offs;\r
   int pcm_mixbuf[PCM_MIXBUF_LEN * 2];\r
   int pcm_mixpos;\r
   char pcm_mixbuf_dirty;\r
index 26a182f..455aac7 100644 (file)
@@ -189,8 +189,8 @@ void PsndRerate(int preserve_state)
   // samples per line (Q16)\r
   Pico.snd.smpl_mult = 65536LL * PicoIn.sndRate / (target_fps*target_lines);\r
   // samples per z80 clock (Q20)\r
-  Pico.snd.clkl_mult = 16 * Pico.snd.smpl_mult * 15/7 / 488.5;\r
-  // samples per 44.1 KHz sample\r
+  Pico.snd.clkz_mult = 16 * Pico.snd.smpl_mult * 15/7 / 488.5;\r
+  // samples per 44.1 KHz sample (Q16)\r
   Pico.snd.cdda_mult = 65536LL * 44100 / PicoIn.sndRate;\r
   Pico.snd.cdda_div  = 65536LL * PicoIn.sndRate / 44100;\r
 \r
@@ -229,7 +229,7 @@ PICO_INTERNAL void PsndDoDAC(int cyc_to)
   if (!PicoIn.sndOut) return;\r
 \r
   // number of samples to fill in buffer (Q20)\r
-  len = (cyc_to * Pico.snd.clkl_mult) - Pico.snd.dac_pos;\r
+  len = (cyc_to * Pico.snd.clkz_mult) - Pico.snd.dac_pos;\r
 \r
   // update position and calculate buffer offset and length\r
   pos = (Pico.snd.dac_pos+0x80000) >> 20;\r
@@ -271,7 +271,7 @@ PICO_INTERNAL void PsndDoPSG(int cyc_to)
   if (!PicoIn.sndOut) return;\r
 \r
   // number of samples to fill in buffer (Q20)\r
-  len = (cyc_to * Pico.snd.clkl_mult) - Pico.snd.psg_pos;\r
+  len = (cyc_to * Pico.snd.clkz_mult) - Pico.snd.psg_pos;\r
 \r
   // update position and calculate buffer offset and length\r
   pos = (Pico.snd.psg_pos+0x80000) >> 20;\r
@@ -301,7 +301,7 @@ PICO_INTERNAL void PsndDoSMSFM(int cyc_to)
   if (!PicoIn.sndOut) return;\r
 \r
   // number of samples to fill in buffer (Q20)\r
-  len = (cyc_to * Pico.snd.clkl_mult) - Pico.snd.ym2413_pos;\r
+  len = (cyc_to * Pico.snd.clkz_mult) - Pico.snd.ym2413_pos;\r
 \r
   // update position and calculate buffer offset and length\r
   pos = (Pico.snd.ym2413_pos+0x80000) >> 20;\r
@@ -342,7 +342,7 @@ PICO_INTERNAL void PsndDoFM(int cyc_to)
   if (!PicoIn.sndOut) return;\r
 \r
   // Q20, number of samples since last call\r
-  len = (cyc_to * Pico.snd.clkl_mult) - Pico.snd.fm_pos;\r
+  len = (cyc_to * Pico.snd.clkz_mult) - Pico.snd.fm_pos;\r
 \r
   // update position and calculate buffer offset and length\r
   pos = (Pico.snd.fm_pos+0x80000) >> 20;\r
@@ -369,7 +369,7 @@ PICO_INTERNAL void PsndDoPCM(int cyc_to)
   if (!PicoIn.sndOut) return;\r
 \r
   // Q20, number of samples since last call\r
-  len = (cyc_to * Pico.snd.clkl_mult) - Pico.snd.pcm_pos;\r
+  len = (cyc_to * Pico.snd.clkz_mult) - Pico.snd.pcm_pos;\r
 \r
   // update position and calculate buffer offset and length\r
   pos = (Pico.snd.pcm_pos+0x80000) >> 20;\r
@@ -391,13 +391,23 @@ static void cdda_raw_update(s32 *buffer, int length, int stereo)
 {\r
   int ret, cdda_bytes;\r
 \r
+  // apply start offset in frame (offset to 1st lba to play)\r
+  int offs = Pico_mcd->cdda_frame_offs * Pico.snd.cdda_div >> 16;\r
+  length -= offs;\r
+  buffer += offs * (stereo ? 2 : 1);\r
+  Pico_mcd->cdda_frame_offs = 0;\r
+\r
   cdda_bytes = (length * Pico.snd.cdda_mult >> 16) * 4;\r
 \r
+  // compute offset of last played sample in this frame (need for save/load)\r
+  Pico_mcd->m.cdda_lba_offset += cdda_bytes/4;\r
+  while (Pico_mcd->m.cdda_lba_offset >= 2352/4)\r
+    Pico_mcd->m.cdda_lba_offset -= 2352/4;\r
+\r
   ret = pm_read_audio(cdda_out_buffer, cdda_bytes, Pico_mcd->cdda_stream);\r
   if (ret < cdda_bytes) {\r
     memset((char *)cdda_out_buffer + ret, 0, cdda_bytes - ret);\r
     Pico_mcd->cdda_stream = NULL;\r
-    return;\r
   }\r
 \r
   // now mix\r
@@ -423,7 +433,9 @@ void cdda_start_play(int lba_base, int lba_offset, int lb_len)
     return;\r
   }\r
 \r
-  pm_seek(Pico_mcd->cdda_stream, (lba_base + lba_offset) * 2352, SEEK_SET);\r
+  // on restart after loading, consider offset of last played sample\r
+  pm_seek(Pico_mcd->cdda_stream, (lba_base + lba_offset) * 2352 +\r
+                                  Pico_mcd->m.cdda_lba_offset * 4, SEEK_SET);\r
   if (Pico_mcd->cdda_type == CT_WAV)\r
   {\r
     // skip headers, assume it's 44kHz stereo uncompressed\r