From 4d97b0a774499c66ed26969ed82c1580fee50412 Mon Sep 17 00:00:00 2001 From: kub Date: Mon, 24 Mar 2025 23:12:32 +0100 Subject: [PATCH] sms, fix ym2413 fm unit sound load/save --- pico/pico_int.h | 3 - pico/sound/sn76496.h | 2 + pico/sound/sound.c | 38 +----------- pico/sound/ym2413.c | 124 +++++++++++++++++++++++++++++++++++++ pico/sound/ym2413.h | 14 +++++ pico/state.c | 22 +++---- platform/common/common.mak | 2 +- 7 files changed, 153 insertions(+), 52 deletions(-) create mode 100644 pico/sound/ym2413.c create mode 100644 pico/sound/ym2413.h diff --git a/pico/pico_int.h b/pico/pico_int.h index aa5fcf3c..c98a02a6 100644 --- a/pico/pico_int.h +++ b/pico/pico_int.h @@ -936,9 +936,6 @@ void ym2612_unpack_state_old(void); Pico.t.timer_b_step = TIMER_B_TICK_ZCYCLES * 256; \ ym2612.OPN.ST.status &= ~3; -void *YM2413GetRegs(void); -void YM2413UnpackState(void); - // videoport.c extern u32 SATaddr, SATmask; static __inline void UpdateSAT(u32 a, u32 d) diff --git a/pico/sound/sn76496.h b/pico/sound/sn76496.h index 8677ea93..af27d3d2 100644 --- a/pico/sound/sn76496.h +++ b/pico/sound/sn76496.h @@ -1,6 +1,8 @@ #ifndef SN76496_H #define SN76496_H +extern int *sn76496_regs; + void SN76496Write(int data); void SN76496Update(short *buffer,int length,int stereo); void SN76496Config(int panning); diff --git a/pico/sound/sound.c b/pico/sound/sound.c index 6a627408..26a182f8 100644 --- a/pico/sound/sound.c +++ b/pico/sound/sound.c @@ -11,8 +11,8 @@ #include #include "../pico_int.h" #include "ym2612.h" +#include "ym2413.h" #include "sn76496.h" -#include "emu2413/emu2413.h" #include "../cd/megasd.h" #include "resampler.h" #include "mix.h" @@ -28,46 +28,10 @@ static s32 PsndBuffer[2*(54000+100)/50+2]; // cdda output buffer s16 cdda_out_buffer[2*1152]; -// sn76496 -extern int *sn76496_regs; - // FM resampling polyphase FIR static resampler_t *fmresampler; static int (*PsndFMUpdate)(s32 *buffer, int length, int stereo, int is_buf_empty); -// ym2413 -static OPLL *opll = NULL; -static struct { - uint32_t adr; - uint8_t reg[sizeof(opll->reg)]; -} opll_buf; - - -void YM2413_regWrite(unsigned data){ - OPLL_writeIO(opll,0,data); -} -void YM2413_dataWrite(unsigned data){ - OPLL_writeIO(opll,1,data); -} - -PICO_INTERNAL void *YM2413GetRegs(void) -{ - memcpy(opll_buf.reg, opll->reg, sizeof(opll->reg)); - opll_buf.adr = opll->adr; - return &opll_buf; -} - -PICO_INTERNAL void YM2413UnpackState(void) -{ - int i; - - for (i = sizeof(opll->reg)-1; i >= 0; i--) { - OPLL_writeIO(opll, 0, i); - OPLL_writeIO(opll, 1, opll_buf.reg[i]); - } - opll->adr = opll_buf.adr; -} - PICO_INTERNAL void PsndInit(void) { opll = OPLL_new(OSC_NTSC/15, OSC_NTSC/15/72); diff --git a/pico/sound/ym2413.c b/pico/sound/ym2413.c new file mode 100644 index 00000000..06ccd2c6 --- /dev/null +++ b/pico/sound/ym2413.c @@ -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 index 00000000..41c46d19 --- /dev/null +++ b/pico/sound/ym2413.h @@ -0,0 +1,14 @@ +/* PicoDrive's wrapper for emu2413 + */ + +#include +#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); diff --git a/pico/state.c b/pico/state.c index 72129139..684610d9 100644 --- a/pico/state.c +++ b/pico/state.c @@ -11,13 +11,11 @@ #include #include "sound/ym2612.h" -#include "sound/emu2413/emu2413.h" +#include "sound/ym2413.h" +#include "sound/sn76496.h" #include "cd/megasd.h" #include "state.h" -// sn76496 & ym2413 -extern int *sn76496_regs; - static arearw *areaRead; static arearw *areaWrite; static areaeof *areaEof; @@ -231,7 +229,6 @@ static int state_save(void *file) { char sbuff[32] = "Saving.. "; unsigned char buff[0x60], buff_z80[Z80_STATE_SIZE]; - void *ym_regs = YM2612GetRegs(); void *buf2 = NULL; int ver = 0x0191; // not really used.. int retval = -1; @@ -262,8 +259,8 @@ static int state_save(void *file) CHECKED_WRITE(CHUNK_PICO, sizeof(PicoPicohw), &PicoPicohw); } else { #ifdef __GP2X__ + void *ym_regs = YM2612GetRegs(); ym2612_pack_state_old(); - ym_regs = YM2612GetRegs(); CHECKED_WRITE(CHUNK_FM, 0x200+4, ym_regs); #else // write fm state first since timer load needs OPN.ST.mode @@ -279,8 +276,11 @@ static int state_save(void *file) } else { CHECKED_WRITE_BUFF(CHUNK_SMS, Pico.ms); - ym_regs = YM2413GetRegs(); - CHECKED_WRITE(CHUNK_YM2413, 0x40+4, ym_regs); + // only store the FM unit state if it was really used + if (Pico.m.hardware & PMS_HW_FMUSED) { + len = ym2413_pack_state(buf2, CHUNK_LIMIT_W); + CHECKED_WRITE(CHUNK_YM2413, len, buf2); + } } CHECKED_WRITE(CHUNK_PSG, 28*4, sn76496_regs); @@ -485,9 +485,9 @@ static int state_load(void *file) case CHUNK_IOPORTS: CHECKED_READ_BUFF(PicoMem.ioports); break; case CHUNK_PSG: CHECKED_READ2(28*4, sn76496_regs); break; case CHUNK_YM2413: - ym_regs = YM2413GetRegs(); - CHECKED_READ2(0x40+4, ym_regs); - YM2413UnpackState(); + CHECKED_READ(len, buf); + ym2413_unpack_state(buf, len); + Pico.m.hardware |= PMS_HW_FMUSED; break; case CHUNK_FM: ym_regs = YM2612GetRegs(); diff --git a/platform/common/common.mak b/platform/common/common.mak index f3d547df..a5abac7e 100644 --- a/platform/common/common.mak +++ b/platform/common/common.mak @@ -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 -- 2.39.5