sms, fix ym2413 fm unit sound load/save
authorkub <derkub@gmail.com>
Mon, 24 Mar 2025 22:12:32 +0000 (23:12 +0100)
committerkub <derkub@gmail.com>
Mon, 24 Mar 2025 22:13:21 +0000 (23:13 +0100)
pico/pico_int.h
pico/sound/sn76496.h
pico/sound/sound.c
pico/sound/ym2413.c [new file with mode: 0644]
pico/sound/ym2413.h [new file with mode: 0644]
pico/state.c
platform/common/common.mak

index aa5fcf3..c98a02a 100644 (file)
@@ -936,9 +936,6 @@ void ym2612_unpack_state_old(void);
   Pico.t.timer_b_step = TIMER_B_TICK_ZCYCLES * 256; \\r
   ym2612.OPN.ST.status &= ~3;\r
 \r
-void *YM2413GetRegs(void);\r
-void YM2413UnpackState(void);\r
-\r
 // videoport.c\r
 extern u32 SATaddr, SATmask;\r
 static __inline void UpdateSAT(u32 a, u32 d)\r
index 8677ea9..af27d3d 100644 (file)
@@ -1,6 +1,8 @@
 #ifndef SN76496_H\r
 #define SN76496_H\r
 \r
+extern int *sn76496_regs;\r
+\r
 void SN76496Write(int data);\r
 void SN76496Update(short *buffer,int length,int stereo);\r
 void SN76496Config(int panning);\r
index 6a62740..26a182f 100644 (file)
@@ -11,8 +11,8 @@
 #include <string.h>\r
 #include "../pico_int.h"\r
 #include "ym2612.h"\r
+#include "ym2413.h"\r
 #include "sn76496.h"\r
-#include "emu2413/emu2413.h"\r
 #include "../cd/megasd.h"\r
 #include "resampler.h"\r
 #include "mix.h"\r
@@ -28,46 +28,10 @@ static s32 PsndBuffer[2*(54000+100)/50+2];
 // cdda output buffer\r
 s16 cdda_out_buffer[2*1152];\r
 \r
-// sn76496\r
-extern int *sn76496_regs;\r
-\r
 // FM resampling polyphase FIR\r
 static resampler_t *fmresampler;\r
 static int (*PsndFMUpdate)(s32 *buffer, int length, int stereo, int is_buf_empty);\r
 \r
-// ym2413\r
-static OPLL *opll = NULL;\r
-static struct {\r
-  uint32_t adr;\r
-  uint8_t reg[sizeof(opll->reg)];\r
-} opll_buf;\r
-\r
-\r
-void YM2413_regWrite(unsigned data){\r
-  OPLL_writeIO(opll,0,data);\r
-}\r
-void YM2413_dataWrite(unsigned data){\r
-  OPLL_writeIO(opll,1,data);\r
-}\r
-\r
-PICO_INTERNAL void *YM2413GetRegs(void)\r
-{\r
-  memcpy(opll_buf.reg, opll->reg, sizeof(opll->reg));\r
-  opll_buf.adr = opll->adr;\r
-  return &opll_buf;\r
-}\r
-\r
-PICO_INTERNAL void YM2413UnpackState(void)\r
-{\r
-  int i;\r
-\r
-  for (i = sizeof(opll->reg)-1; i >= 0; i--) {\r
-    OPLL_writeIO(opll, 0, i);\r
-    OPLL_writeIO(opll, 1, opll_buf.reg[i]);\r
-  }\r
-  opll->adr = opll_buf.adr;\r
-}\r
-\r
 PICO_INTERNAL void PsndInit(void)\r
 {\r
   opll = OPLL_new(OSC_NTSC/15, OSC_NTSC/15/72);\r
diff --git a/pico/sound/ym2413.c b/pico/sound/ym2413.c
new file mode 100644 (file)
index 0000000..06ccd2c
--- /dev/null
@@ -0,0 +1,124 @@
+/* PicoDrive's wrapper for emu2413
+ */
+
+#include "emu2413/emu2413.c"
+
+// the one instance that can be in a Mark III
+OPLL *opll = NULL;
+
+
+void YM2413_regWrite(unsigned data){
+  OPLL_writeIO(opll,0,data);
+}
+
+void YM2413_dataWrite(unsigned data){
+  OPLL_writeIO(opll,1,data);
+}
+
+
+// state saving/loading - old save states only have adr and reg saved, new
+// saves have necessary internal state data as well. Most of the state is
+// recovered from the registers, which keeps the internal state data smaller.
+
+#include "../state.h"
+
+#define SLOT_SIZE_MIN 12
+#define OTHER_SIZE_MIN 32
+
+static size_t save_slot(u8 *buf, const OPLL_SLOT *slot)
+{
+       size_t b = 0;
+
+       b++; // length, assumes slot state won't grow beyond 255
+       save_u32(buf, &b, slot->pg_phase);
+       save_u8_(buf, &b, slot->key_flag);
+       save_u8_(buf, &b, slot->eg_state);
+       save_u8_(buf, &b, slot->eg_out);
+       save_s16(buf, &b, slot->output[0]);
+       save_s16(buf, &b, slot->output[1]);
+
+       //printf("slot size: %zd\n", b);
+       assert(b >= SLOT_SIZE_MIN);
+       assert(b < 256u);
+       buf[0] = b - 1;
+       return b;
+}
+
+static void load_slot(const u8 *buf, OPLL_SLOT *slot)
+{
+       size_t b = 0;
+
+       slot->pg_phase  = load_u32(buf, &b);
+       slot->key_flag  = load_u8_(buf, &b);
+       slot->eg_state  = load_u8_(buf, &b);
+       slot->eg_out    = load_u8_(buf, &b);
+       slot->output[0] = load_s16(buf, &b);
+       slot->output[1] = load_s16(buf, &b);
+
+       slot->pg_out = slot->pg_phase >> DP_BASE_BITS;
+}
+
+size_t ym2413_pack_state(void *buf_, size_t size)
+{
+       size_t i, b = 0;
+       u8 *buf = buf_;
+
+       // regs and adr first, for backwards compatibility
+       save_u32(buf, &b, opll->adr);
+       for (i = 0; i < 0x40; i++)
+               save_u8_(buf, &b, opll->reg[i]);
+
+       // user patches only, all others are read-only anyway
+       for (i = 0; i < 18; i++)
+               b += save_slot(&buf[b], &opll->slot[i]);
+
+       for (i = 0; i < 9; i++)
+               save_u8_(buf, &b, opll->patch_number[i]);
+       save_u32(buf, &b, opll->slot_key_status);
+       save_u32(buf, &b, opll->eg_counter);
+       save_u8_(buf, &b, opll->lfo_am);
+       save_u32(buf, &b, opll->pm_phase);
+       save_s32(buf, &b, opll->am_phase);
+       save_u32(buf, &b, opll->noise);
+       save_u16(buf, &b, opll->short_noise);
+
+       printf("ym2413 state size: %zu\n", b);
+       assert(b <= size);
+       return b;
+}
+
+void ym2413_unpack_state(const void *buf_, size_t size)
+{
+       const u8 *buf = buf_;
+       size_t i, b = 0;
+
+       // registers, write to opll too to take over to internal data structures
+       opll->adr = load_u32(buf, &b);
+       for (i = 0; i < 0x40; i++) {
+               opll->reg[i] = load_u8_(buf, &b);
+               // skip the shadow registers
+               if ((i & 0xf) < 9 || (i & 0x30) == 0)
+                       OPLL_writeReg(opll, i, opll->reg[i]);
+       }
+
+       if (b >= size) return; // old save
+
+       for (i = 0; i < 18; i++) {
+               u8 sz = load_u8_(buf, &b);
+               load_slot(&buf[b], &opll->slot[i]);
+               b += sz;
+       }
+
+       for (i = 0; i < 9; i++)
+               opll->patch_number[i] = load_u8_(buf, &b);
+       opll->slot_key_status = load_u32(buf, &b);
+       opll->eg_counter =  load_u32(buf, &b);
+       opll->lfo_am =      load_u8_(buf, &b);
+       opll->pm_phase =    load_u32(buf, &b);
+       opll->am_phase =    load_s32(buf, &b);
+       opll->noise =       load_u32(buf, &b);
+       opll->short_noise = load_u16(buf, &b);
+
+       OPLL_forceRefresh(opll);
+}
+
diff --git a/pico/sound/ym2413.h b/pico/sound/ym2413.h
new file mode 100644 (file)
index 0000000..41c46d1
--- /dev/null
@@ -0,0 +1,14 @@
+/* PicoDrive's wrapper for emu2413
+ */
+
+#include <stddef.h>
+#include "emu2413/emu2413.h"
+
+// the one instance that can be in a Mark III
+extern OPLL *opll;
+
+void YM2413_regWrite(unsigned data);
+void YM2413_dataWrite(unsigned data);
+
+size_t ym2413_pack_state(void *buf_, size_t size);
+void ym2413_unpack_state(const void *buf_, size_t size);
index 7212913..684610d 100644 (file)
 \r
 #include <cpu/sh2/sh2.h>\r
 #include "sound/ym2612.h"\r
-#include "sound/emu2413/emu2413.h"\r
+#include "sound/ym2413.h"\r
+#include "sound/sn76496.h"\r
 #include "cd/megasd.h"\r
 #include "state.h"\r
 \r
-// sn76496 & ym2413\r
-extern int *sn76496_regs;\r
-\r
 static arearw    *areaRead;\r
 static arearw    *areaWrite;\r
 static areaeof   *areaEof;\r
@@ -231,7 +229,6 @@ static int state_save(void *file)
 {\r
   char sbuff[32] = "Saving.. ";\r
   unsigned char buff[0x60], buff_z80[Z80_STATE_SIZE];\r
-  void *ym_regs = YM2612GetRegs();\r
   void *buf2 = NULL;\r
   int ver = 0x0191; // not really used..\r
   int retval = -1;\r
@@ -262,8 +259,8 @@ static int state_save(void *file)
       CHECKED_WRITE(CHUNK_PICO, sizeof(PicoPicohw), &PicoPicohw);\r
     } else {\r
 #ifdef __GP2X__\r
+      void *ym_regs = YM2612GetRegs();\r
       ym2612_pack_state_old();\r
-      ym_regs = YM2612GetRegs();\r
       CHECKED_WRITE(CHUNK_FM, 0x200+4, ym_regs);\r
 #else\r
       // write fm state first since timer load needs OPN.ST.mode\r
@@ -279,8 +276,11 @@ static int state_save(void *file)
   }\r
   else {\r
     CHECKED_WRITE_BUFF(CHUNK_SMS, Pico.ms);\r
-    ym_regs = YM2413GetRegs();\r
-    CHECKED_WRITE(CHUNK_YM2413, 0x40+4, ym_regs);\r
+    // only store the FM unit state if it was really used\r
+    if (Pico.m.hardware & PMS_HW_FMUSED) {\r
+      len = ym2413_pack_state(buf2, CHUNK_LIMIT_W);\r
+      CHECKED_WRITE(CHUNK_YM2413, len, buf2);\r
+    }\r
   }\r
   CHECKED_WRITE(CHUNK_PSG, 28*4, sn76496_regs);\r
 \r
@@ -485,9 +485,9 @@ static int state_load(void *file)
       case CHUNK_IOPORTS: CHECKED_READ_BUFF(PicoMem.ioports); break;\r
       case CHUNK_PSG:     CHECKED_READ2(28*4, sn76496_regs); break;\r
       case CHUNK_YM2413:\r
-        ym_regs = YM2413GetRegs();\r
-        CHECKED_READ2(0x40+4, ym_regs);\r
-        YM2413UnpackState();\r
+        CHECKED_READ(len, buf);\r
+        ym2413_unpack_state(buf, len);\r
+        Pico.m.hardware |= PMS_HW_FMUSED;\r
         break;\r
       case CHUNK_FM:\r
         ym_regs = YM2612GetRegs();\r
index f3d547d..a5abac7 100644 (file)
@@ -124,7 +124,7 @@ endif
 # sound
 SRCS_COMMON += $(R)pico/sound/sound.c $(R)pico/sound/resampler.c
 SRCS_COMMON += $(R)pico/sound/sn76496.c $(R)pico/sound/ym2612.c
-SRCS_COMMON += $(R)pico/sound/emu2413/emu2413.c
+SRCS_COMMON += $(R)pico/sound/ym2413.c
 ifneq "$(ARCH)$(asm_mix)" "arm1"
 SRCS_COMMON += $(R)pico/sound/mix.c
 endif