From ccce7ee7bfae9246ea1dec0d866ce8e15af5babc Mon Sep 17 00:00:00 2001 From: notaz Date: Fri, 7 Mar 2025 02:28:47 +0200 Subject: [PATCH] sound: new ym2612 save states irixxxx/picodrive#166 --- pico/memory.c | 46 ++++++-- pico/pico_int.h | 5 +- pico/sound/sound.c | 14 ++- pico/sound/ym2612.c | 258 +++++++++++++++++++++++++++++++++++++++++--- pico/sound/ym2612.h | 18 ++-- pico/state.c | 20 +++- pico/state.h | 69 ++++++++++++ 7 files changed, 391 insertions(+), 39 deletions(-) diff --git a/pico/memory.c b/pico/memory.c index 4ed955c7..87b847f2 100644 --- a/pico/memory.c +++ b/pico/memory.c @@ -8,8 +8,10 @@ * See COPYING file in the top-level directory. */ +#include #include "pico_int.h" #include "memory.h" +#include "state.h" #include "sound/ym2612.h" #include "sound/sn76496.h" @@ -1327,7 +1329,7 @@ static int ym2612_write_local(u32 a, u32 d, int is_from_z80) //elprintf(EL_STATUS, "%03i dac w %08x z80 %i", cycles, d, is_from_z80); if (ym2612.dacen) PsndDoDAC(cycles); - ym2612.dacout = ((int)d - 0x80) << 6; + ym2612.dacout = ((int)d - 0x80) << DAC_SHIFT; return 0; } case 0x2b: { /* DAC Sel (YM2612) */ @@ -1374,10 +1376,11 @@ static u32 ym2612_read_local_68k(void) return ym2612.OPN.ST.status; } -void ym2612_pack_state(void) +int ym2612_pack_timers(void *buf, size_t size) { // timers are saved as tick counts, in 16.16 int format int tac, tat = 0, tbc, tbt = 0, busy = 0; + size_t b = 0; tac = 1024 - ym2612.OPN.ST.TA; tbc = 256 - ym2612.OPN.ST.TB; if (Pico.t.ym2612_busy > 0) @@ -1388,18 +1391,26 @@ void ym2612_pack_state(void) tbt = ((Pico.t.timer_b_step - Pico.t.timer_b_next_oflow) * ((1LL<<32)/TIMER_B_TICK_ZCYCLES+1))>>16; elprintf(EL_YMTIMER, "save: timer a %i/%i", tat >> 16, tac); elprintf(EL_YMTIMER, "save: timer b %i/%i", tbt >> 16, tbc); - #ifdef __GP2X__ if (PicoIn.opt & POPT_EXT_FM) YM2612PicoStateSave2_940(tat, tbt); else #endif - YM2612PicoStateSave2(tat, tbt, busy); + { + //YM2612PicoStateSave2(tat, tbt, busy); + assert(size >= 16); + save_u16(buf, &b, ym2612.OPN.ST.TA); + save_u16(buf, &b, ym2612.OPN.ST.TB); + save_u32(buf, &b, tat); + save_u32(buf, &b, tbt); + save_u32(buf, &b, busy); + } + return b; } -void ym2612_unpack_state(void) +void ym2612_unpack_state_old(void) { - int i, ret, tac, tat, tbc, tbt, busy = 0; + int i, ret, tat, tbt, busy = 0; YM2612PicoStateLoad(); Pico.t.m68c_frame_start = SekCyclesDone(); @@ -1435,7 +1446,30 @@ void ym2612_unpack_state(void) elprintf(EL_STATUS, "old ym2612 state"); return; // no saved timers } + { + u8 tmp[16]; + size_t b = 0; + save_u16(tmp, &b, ym2612.OPN.ST.TA); + save_u16(tmp, &b, ym2612.OPN.ST.TB); + save_u32(tmp, &b, tat); + save_u32(tmp, &b, tbt); + save_u32(tmp, &b, busy); + ym2612_unpack_timers(tmp, b); + } +} +void ym2612_unpack_timers(const void *buf, size_t size) +{ + int tac, tat, tbc, tbt, busy; + size_t b = 0; + assert(size >= 16); + if (size < 16) + return; + ym2612.OPN.ST.TA = load_u16(buf, &b); + ym2612.OPN.ST.TB = load_u16(buf, &b); + tat = load_u32(buf, &b); + tbt = load_u32(buf, &b); + busy = load_u32(buf, &b); Pico.t.ym2612_busy = cycles_68k_to_z80(busy); tac = (1024 - ym2612.OPN.ST.TA) << 16; tbc = (256 - ym2612.OPN.ST.TB) << 16; diff --git a/pico/pico_int.h b/pico/pico_int.h index 02aa468e..612783a6 100644 --- a/pico/pico_int.h +++ b/pico/pico_int.h @@ -904,8 +904,9 @@ void cdda_start_play(int lba_base, int lba_offset, int lb_len); #define YM2612_NATIVE_RATE() (((Pico.m.pal?OSC_PAL:OSC_NTSC)/7 + 3*24) / (6*24)) void ym2612_sync_timers(int z80_cycles, int mode_old, int mode_new); -void ym2612_pack_state(void); -void ym2612_unpack_state(void); +int ym2612_pack_timers(void *buf_, size_t size); +void ym2612_unpack_timers(const void *buf_, size_t size); +void ym2612_unpack_state_old(void); #define TIMER_NO_OFLOW 0x70000000 diff --git a/pico/sound/sound.c b/pico/sound/sound.c index d7de6e9d..6a627408 100644 --- a/pico/sound/sound.c +++ b/pico/sound/sound.c @@ -170,6 +170,7 @@ void PsndRerate(int preserve_state) int ym2612_clock = Pico.m.pal ? OSC_PAL/7 : OSC_NTSC/7; int ym2612_rate = YM2612_NATIVE_RATE(); int ym2612_init = !preserve_state; + int state_size = 4096; // don't init YM2612 if preserve_state and no parameter changes ym2612_init |= ymclock != ym2612_clock || ymopts != (PicoIn.opt & (POPT_DIS_FM_SSGEG|POPT_EN_FM_DAC)); @@ -179,10 +180,9 @@ void PsndRerate(int preserve_state) ymopts = PicoIn.opt & (POPT_DIS_FM_SSGEG|POPT_EN_FM_DAC); if (preserve_state && ym2612_init) { - state = malloc(0x204); - if (state == NULL) return; - ym2612_pack_state(); - memcpy(state, YM2612GetRegs(), 0x204); + state = malloc(state_size); + if (state) + state_size = YM2612PicoStateSave3(state, state_size); } if (PicoIn.AHW & PAHW_SMS) { @@ -207,10 +207,8 @@ void PsndRerate(int preserve_state) PsndFMUpdate = YM2612UpdateONE; } - if (preserve_state && ym2612_init) { - // feed it back it's own registers, just like after loading state - memcpy(YM2612GetRegs(), state, 0x204); - ym2612_unpack_state(); + if (state) { + YM2612PicoStateLoad3(state, state_size); free(state); } diff --git a/pico/sound/ym2612.c b/pico/sound/ym2612.c index efa2c10b..3639b018 100644 --- a/pico/sound/ym2612.c +++ b/pico/sound/ym2612.c @@ -111,6 +111,7 @@ //#include +#include #include #include @@ -153,7 +154,6 @@ void memset32(void *dest, int c, int count); #define FREQ_SH 16 /* 16.16 fixed point (frequency calculations) */ #define EG_SH 16 /* 16.16 fixed point (envelope generator timing) */ #define LFO_SH 24 /* 8.24 fixed point (LFO calculations) */ -#define TIMER_SH 16 /* 16.16 fixed point (timers calculations) */ #define ENV_BITS 10 #define ENV_LEN (1<increment counter */ -static int g_lfo_ampm; - /* register number to channel number , slot offset */ #define OPN_CHAN(N) (N&3) #define OPN_SLOT(N) ((N>>2)&3) @@ -1228,7 +1226,7 @@ static chan_rend_context crct; static void chan_render_prep(void) { crct.eg_timer_add = ym2612.OPN.eg_timer_add; - crct.lfo_init_sft16 = g_lfo_ampm << 16; + crct.lfo_init_sft16 = ym2612.OPN.lfo_ampm << 16; crct.lfo_inc = ym2612.OPN.lfo_inc; } @@ -1437,9 +1435,9 @@ static void reset_channels(FM_CH *CH) ym2612.OPN.ST.mode = 0; /* normal mode */ ym2612.OPN.ST.TA = 0; - ym2612.OPN.ST.TAC = 0; + //ym2612.OPN.ST.TAC = 0; ym2612.OPN.ST.TB = 0; - ym2612.OPN.ST.TBC = 0; + //ym2612.OPN.ST.TBC = 0; for( c = 0 ; c < 6 ; c++ ) { @@ -1815,7 +1813,7 @@ int YM2612UpdateOne_(s32 *buffer, int length, int stereo, int is_buf_empty) if (ym2612.slot_mask & 0x00f000) active_chs |= chan_render(buffer, length, 3, flags|((pan&0x0c0)>>2)) << 3; BIT_IF(flags, 1, (ym2612.ssg_mask & 0x0f0000) && (ym2612.OPN.ST.flags & 1)); if (ym2612.slot_mask & 0x0f0000) active_chs |= chan_render(buffer, length, 4, flags|((pan&0x300)>>4)) << 4; - g_lfo_ampm = crct.pack >> 16; // need_save; now because ch5 might skip updating it + ym2612.OPN.lfo_ampm = crct.pack >> 16; // need_save; now because ch5 might skip updating it BIT_IF(flags, 1, (ym2612.ssg_mask & 0xf00000) && (ym2612.OPN.ST.flags & 1)); if (ym2612.slot_mask & 0xf00000) active_chs |= chan_render(buffer, length, 5, flags|((pan&0xc00)>>6)|(!!ym2612.dacen<<2)) << 5; #undef BIT_IF @@ -1856,7 +1854,7 @@ void YM2612ResetChip_(void) ym2612.OPN.eg_cnt = 0; ym2612.OPN.lfo_inc = 0; ym2612.OPN.lfo_cnt = 0; - g_lfo_ampm = 126 << 8; + ym2612.OPN.lfo_ampm = 126 << 8; ym2612.OPN.ST.status = 0; reset_channels( &ym2612.CH[0] ); @@ -1917,7 +1915,7 @@ int YM2612Write_(unsigned int a, unsigned int v) { ym2612.OPN.lfo_inc = 0; ym2612.OPN.lfo_cnt = 0; - g_lfo_ampm = 126 << 8; + ym2612.OPN.lfo_ampm = 126 << 8; } break; #if 0 // handled elsewhere @@ -1970,7 +1968,7 @@ int YM2612Write_(unsigned int a, unsigned int v) break; } case 0x2a: /* DAC data (YM2612) */ - ym2612.dacout = ((int)v - 0x80) << 6; /* level unknown (notaz: 8 seems to be too much) */ + ym2612.dacout = ((int)v - 0x80) << DAC_SHIFT; ret=0; break; case 0x2b: /* DAC Sel (YM2612) */ @@ -2030,6 +2028,7 @@ void YM2612PicoStateLoad_(void) } /* rather stupid design because I wanted to fit in unused register "space" */ +// TODO remove all this along with ym2612.REGS typedef struct { UINT32 state_phase; @@ -2064,7 +2063,6 @@ typedef struct #define _block_fnum op1_out_l #define _block_fnum_sl3 unused_sl3 - void YM2612PicoStateSave2(int tat, int tbt, int busy) { ym_save_addon_slot ss; @@ -2140,7 +2138,7 @@ void YM2612PicoStateSave2(int tat, int tbt, int busy) sa.eg_cnt = ym2612.OPN.eg_cnt; sa.eg_timer = ym2612.OPN.eg_timer; sa.lfo_cnt = ym2612.OPN.lfo_cnt; - sa.lfo_ampm = g_lfo_ampm; + sa.lfo_ampm = ym2612.OPN.lfo_ampm; sa.busy_timer = busy; //sa.keyon_field = ym2612.slot_mask; memcpy(ptr, &sa, sizeof(sa)); // 0x30 max @@ -2168,7 +2166,7 @@ int YM2612PicoStateLoad2(int *tat, int *tbt, int *busy) ym2612.OPN.eg_cnt = sa.eg_cnt; ym2612.OPN.eg_timer = sa.eg_timer; ym2612.OPN.lfo_cnt = sa.lfo_cnt; - g_lfo_ampm = sa.lfo_ampm; + ym2612.OPN.lfo_ampm = sa.lfo_ampm; ym2612.slot_mask = sa.keyon_field; if (tat != NULL) *tat = sa.TAT; if (tbt != NULL) *tbt = sa.TBT; @@ -2247,6 +2245,240 @@ int YM2612PicoStateLoad2(int *tat, int *tbt, int *busy) return 0; } +#include "../state.h" + +#define SLOT_SIZE_MIN 46 +#define CH_SIZE_MIN 16 +#define OTHER_SIZE_MIN 35 + +static size_t save_slot(u8 *buf, const FM_SLOT *slot) +{ + size_t tmp, b = 0; + + b++; // length, assumes slot state won't grow beyond 255 + tmp = (slot->DT - ym2612.OPN.ST.dt_tab[0]) / sizeof(ym2612.OPN.ST.dt_tab[0]); + save_u8_(buf, &b, tmp); + save_u8_(buf, &b, slot->ar); + save_u8_(buf, &b, slot->d1r); + save_u8_(buf, &b, slot->d2r); + save_u8_(buf, &b, slot->rr); + save_u8_(buf, &b, slot->mul); + save_u32(buf, &b, slot->phase); + save_u32(buf, &b, slot->Incr); + save_u8_(buf, &b, slot->KSR); + save_u8_(buf, &b, slot->ksr); + save_u8_(buf, &b, slot->key); + save_u8_(buf, &b, slot->state); + save_u8_(buf, &b, slot->tl >> (ENV_BITS-7)); + save_u16(buf, &b, slot->volume); + save_u32(buf, &b, slot->sl); + save_u32(buf, &b, slot->eg_pack_rr); + save_u32(buf, &b, slot->eg_pack_d2r); + save_u32(buf, &b, slot->eg_pack_d1r); + save_u32(buf, &b, slot->eg_pack_ar); + save_u8_(buf, &b, slot->ssg); + save_u8_(buf, &b, slot->ssgn); + save_u8_(buf, &b, slot->ar_ksr); + save_u16(buf, &b, slot->vol_out); + + //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, FM_SLOT *slot) +{ + size_t b = 0; + u8 dt_reg; + + dt_reg = load_u8_(buf, &b); + slot->ar = load_u8_(buf, &b); + slot->d1r = load_u8_(buf, &b); + slot->d2r = load_u8_(buf, &b); + slot->rr = load_u8_(buf, &b); + slot->mul = load_u8_(buf, &b); + slot->phase = load_u32(buf, &b); + slot->Incr = load_u32(buf, &b); + slot->KSR = load_u8_(buf, &b); + slot->ksr = load_u8_(buf, &b); + slot->key = load_u8_(buf, &b); + slot->state = load_u8_(buf, &b); + slot->tl = load_u8_(buf, &b) << (ENV_BITS-7); + slot->volume = load_s16(buf, &b); + slot->sl = load_u32(buf, &b); + slot->eg_pack_rr = load_u32(buf, &b); + slot->eg_pack_d2r = load_u32(buf, &b); + slot->eg_pack_d1r = load_u32(buf, &b); + slot->eg_pack_ar = load_u32(buf, &b); + slot->ssg = load_u8_(buf, &b); + slot->ssgn = load_u8_(buf, &b); + slot->ar_ksr = load_u8_(buf, &b); + slot->vol_out = load_u16(buf, &b); + + assert(dt_reg < 8); + slot->DT = ym2612.OPN.ST.dt_tab[dt_reg & 7]; +} + +static size_t save_channel(u8 *buf, const FM_CH *ch) +{ + int i, size_pos; + size_t b = 0; + + for (i = 0; i < 4; i++) + b += save_slot(&buf[b], &ch->SLOT[i]); + size_pos = b++; + save_u8_(buf, &b, ch->ALGO); + save_u8_(buf, &b, ch->FB); + save_u32(buf, &b, ch->op1_out); + save_s16(buf, &b, ch->mem_value); // fits in 16bit + save_u8_(buf, &b, ch->pms); // max 7*32 + save_u8_(buf, &b, ch->ams); + save_u8_(buf, &b, ch->kcode); + save_u8_(buf, &b, ch->upd_cnt); + // ch->fc is derived from .block_fnum + save_u16(buf, &b, ch->block_fnum); + save_u8_(buf, &b, ch->AMmasks); + + assert(b - size_pos - 1 < 256u); + buf[size_pos] = b - size_pos - 1; + return b; +} + +static size_t load_channel(const u8 *buf, size_t size, FM_CH *ch) +{ + size_t i, b = 0, slot_size = 0, ch_size; + u32 fn, blk; + + for (i = 0; i < 4; i++) { + u8 size_next = load_u8_(buf, &slot_size); + if (size_next < SLOT_SIZE_MIN) + return 0; + if (slot_size + size_next > size) + return 0; + load_slot(&buf[slot_size], &ch->SLOT[i]); + slot_size += size_next; + } + if (slot_size + CH_SIZE_MIN > size) + return 0; + b = slot_size; + ch_size = load_u8_(buf, &b); + ch->ALGO = load_u8_(buf, &b); + ch->FB = load_u8_(buf, &b); + ch->op1_out = load_u32(buf, &b); + ch->mem_value = load_s16(buf, &b); + ch->pms = load_u8_(buf, &b); + ch->ams = load_u8_(buf, &b); + ch->kcode = load_u8_(buf, &b); + ch->upd_cnt = load_u8_(buf, &b); + ch->block_fnum = load_u16(buf, &b) & 0x3fff; + ch->AMmasks = load_u8_(buf, &b); + + fn = ch->block_fnum & 0x7ff; + blk = ch->block_fnum >> 11; + ch->fc = fn_table[fn*2] >> (7 - blk); + + assert(ch_size >= b - slot_size - 1); + return slot_size + 1 + ch_size; +} + +size_t YM2612PicoStateSave3(void *buf_, size_t size) +{ + size_t i, b = 0; + u8 *buf = buf_; + u8 lfo_inc_reg = 0; + + for (i = 0; i < 8; i++) { + if (ym2612.OPN.lfo_inc == ym2612.OPN.lfo_freq[i]) { + lfo_inc_reg = i + 1; + break; + } + } + assert(ym2612.OPN.lfo_inc == 0 || i < 8); + + for (i = 0; i < 6; i++) + b += save_channel(&buf[b], &ym2612.CH[i]); + save_u8_(buf, &b, ym2612.OPN.ST.address); + save_u8_(buf, &b, ym2612.OPN.ST.status); + save_u8_(buf, &b, ym2612.OPN.ST.mode); + save_u8_(buf, &b, ym2612.OPN.ST.flags); + // (timers are saved in CHUNK_FM_TIMERS) + save_u8_(buf, &b, ym2612.OPN.ST.fn_h); + save_u8_(buf, &b, ym2612.OPN.SL3.fn_h); + for (i = 0; i < 3; i++) { + // ym2612.OPN.SL3.fc is derived from .block_fnum + save_u8_(buf, &b, ym2612.OPN.SL3.kcode[i]); + save_u16(buf, &b, ym2612.OPN.SL3.block_fnum[i]); + } + save_u16(buf, &b, ym2612.OPN.pan); + save_u16(buf, &b, ym2612.OPN.eg_cnt); + save_u16(buf, &b, ym2612.OPN.eg_timer); + save_u32(buf, &b, ym2612.OPN.lfo_cnt); + save_u16(buf, &b, ym2612.OPN.lfo_ampm); + save_u8_(buf, &b, lfo_inc_reg); + save_u8_(buf, &b, ym2612.addr_A1); + save_u8_(buf, &b, ym2612.dacen); + save_s8_(buf, &b, ym2612.dacout >> DAC_SHIFT); + save_u32(buf, &b, ym2612.ssg_mask); + + //printf("ym2612 state size: %zu\n", b); + assert(b <= size); + return b; +} + +void YM2612PicoStateLoad3(const void *buf_, size_t size) +{ + const u8 *buf = buf_; + size_t i, b = 0; + u8 lfo_inc_reg = 0; + + for (i = 0; i < 6; i++) { + size_t r = load_channel(&buf[b], size - b, &ym2612.CH[i]); + if (!r) + goto broken; + b += r; + } + if (b + OTHER_SIZE_MIN > size) + goto broken; + ym2612.OPN.ST.address = load_u8_(buf, &b); + ym2612.OPN.ST.status = load_u8_(buf, &b); + ym2612.OPN.ST.mode = load_u8_(buf, &b); + ym2612.OPN.ST.flags = load_u8_(buf, &b); + ym2612.OPN.ST.fn_h = load_u8_(buf, &b); + ym2612.OPN.SL3.fn_h = load_u8_(buf, &b); + for (i = 0; i < 3; i++) { + u32 fn, blk; + ym2612.OPN.SL3.kcode[i] = load_u8_(buf, &b); + ym2612.OPN.SL3.block_fnum[i] = load_u16(buf, &b) & 0x3fff; + + fn = ym2612.OPN.SL3.block_fnum[i] & 0x7ff; + blk = ym2612.OPN.SL3.block_fnum[i] >> 11; + ym2612.OPN.SL3.fc[i] = fn_table[fn*2] >> (7 - blk); + } + ym2612.OPN.pan = load_u16(buf, &b); + ym2612.OPN.eg_cnt = load_u16(buf, &b); + ym2612.OPN.eg_timer = load_u16(buf, &b); + ym2612.OPN.lfo_cnt = load_u32(buf, &b); + ym2612.OPN.lfo_ampm = load_u16(buf, &b); + lfo_inc_reg = load_u8_(buf, &b); + ym2612.addr_A1 = load_u8_(buf, &b); + ym2612.dacen = load_u8_(buf, &b); + ym2612.dacout = load_s8_(buf, &b); + ym2612.ssg_mask = load_u32(buf, &b); + + assert(lfo_inc_reg < 9u); + ym2612.OPN.lfo_inc = 0; + if (lfo_inc_reg) + ym2612.OPN.lfo_inc = ym2612.OPN.lfo_freq[--lfo_inc_reg & 7]; + ym2612.dacout = (u32)ym2612.dacout << DAC_SHIFT; + ym2612.slot_mask = 0xffffff; + //printf("ym2612 state size: %zu\n", b); + return; +broken: + elprintf(EL_STATUS, "broken ym2612 state"); +} + void *YM2612GetRegs(void) { return ym2612.REGS; diff --git a/pico/sound/ym2612.h b/pico/sound/ym2612.h index 56ec5ef9..bdf53be8 100644 --- a/pico/sound/ym2612.h +++ b/pico/sound/ym2612.h @@ -5,6 +5,10 @@ #ifndef _H_FM_FM_ #define _H_FM_FM_ +#include + +#define DAC_SHIFT 6 + /* compiler dependence */ #include "../pico_types.h" #ifndef UINT8 @@ -98,13 +102,13 @@ typedef struct UINT8 mode; /* mode CSM / 3SLOT */ UINT8 flags; /* operational flags */ int TA; /* timer a */ - int TAC; /* timer a maxval */ - int TAT; /* timer a ticker | need_save */ + //int TAC; /* timer a maxval */ + //int TAT; /* timer a ticker | need_save */ UINT8 TB; /* timer b */ UINT8 fn_h; /* freq latch */ UINT8 pad2[2]; - int TBC; /* timer b maxval */ - int TBT; /* timer b ticker | need_save */ + //int TBC; /* timer b maxval */ + //int TBT; /* timer b ticker | need_save */ /* local time tables */ INT32 dt_tab[8][32];/* DeTune table */ } FM_ST; @@ -139,6 +143,7 @@ typedef struct /* LFO */ UINT32 lfo_cnt; /* need_save */ UINT32 lfo_inc; + UINT32 lfo_ampm; UINT32 lfo_freq[8]; /* LFO FREQ table */ } FM_OPN; @@ -146,7 +151,7 @@ typedef struct /* here's the virtual YM2612 */ typedef struct { - UINT8 REGS[0x200]; /* registers (for save states) */ + UINT8 REGS[0x200]; /* registers (for old save states) */ INT32 addr_A1; /* address line A1 | need_save */ FM_CH CH[6]; /* channel state */ @@ -177,8 +182,9 @@ int YM2612PicoTick_(int n); void YM2612PicoStateLoad_(void); void *YM2612GetRegs(void); -void YM2612PicoStateSave2(int tat, int tbt, int busy); int YM2612PicoStateLoad2(int *tat, int *tbt, int *busy); +size_t YM2612PicoStateSave3(void *buf_, size_t size); +void YM2612PicoStateLoad3(const void *buf_, size_t size); /* NB must be macros for compiling GP2X 940 code */ #ifndef __GP2X__ diff --git a/pico/state.c b/pico/state.c index 9625621b..e1d7fe16 100644 --- a/pico/state.c +++ b/pico/state.c @@ -138,6 +138,8 @@ typedef enum { CHUNK_PICO, CHUNK_CD_MSD, CHUNK_VDP, + CHUNK_FM_TIMERS, + CHUNK_FMv3, // CHUNK_DEFAULT_COUNT, CHUNK_CARTHW_ = CHUNK_CARTHW, // 64 (defined in PicoInt) @@ -256,9 +258,11 @@ static int state_save(void *file) CHECKED_WRITE(CHUNK_PICO_PCM, len, buf2); CHECKED_WRITE(CHUNK_PICO, sizeof(PicoPicohw), &PicoPicohw); } else { - ym2612_pack_state(); - ym_regs = YM2612GetRegs(); - CHECKED_WRITE(CHUNK_FM, 0x200+4, ym_regs); + // write fm state first since timer load needs OPN.ST.mode + len = YM2612PicoStateSave3(buf2, CHUNK_LIMIT_W); + CHECKED_WRITE(CHUNK_FMv3, len, buf2); + len = ym2612_pack_timers(buf2, CHUNK_LIMIT_W); + CHECKED_WRITE(CHUNK_FM_TIMERS, len, buf2); } if (!(PicoIn.opt & POPT_DIS_IDLE_DET)) @@ -479,7 +483,15 @@ static int state_load(void *file) case CHUNK_FM: ym_regs = YM2612GetRegs(); CHECKED_READ2(0x200+4, ym_regs); - ym2612_unpack_state(); + ym2612_unpack_state_old(); + break; + case CHUNK_FM_TIMERS: + CHECKED_READ(len, buf); + ym2612_unpack_timers(buf, len); + break; + case CHUNK_FMv3: + CHECKED_READ(len, buf); + YM2612PicoStateLoad3(buf, len); break; case CHUNK_PICO_PCM: diff --git a/pico/state.h b/pico/state.h index 1d380223..4f49590f 100644 --- a/pico/state.h +++ b/pico/state.h @@ -1,4 +1,6 @@ #include +#include +#include "pico_types.h" typedef size_t (arearw)(void *p, size_t _size, size_t _n, void *file); typedef size_t (areaeof)(void *file); @@ -7,3 +9,70 @@ typedef int (areaclose)(void *file); int PicoStateFP(void *afile, int is_save, arearw *read, arearw *write, areaeof *eof, areaseek *seek); + +static inline void save_u8_(u8 *buf, size_t *b, u32 u) +{ + assert(!(u & ~0xff)); + buf[(*b)++] = u; +} + +static inline void save_s8_(u8 *buf, size_t *b, s32 s) +{ + s32 s_sext = (s32)((u32)s << 24) >> 24; + assert(s == s_sext); (void)s_sext; + buf[(*b)++] = s; +} + +static inline void save_u16(u8 *buf, size_t *b, u32 u) +{ + assert(!(u & ~0xffff)); + buf[(*b)++] = u; + buf[(*b)++] = u >> 8; +} + +static inline void save_s16(u8 *buf, size_t *b, s32 s) +{ + s32 s_sext = (s32)((u32)s << 16) >> 16; + assert(s == s_sext); (void)s_sext; + buf[(*b)++] = s; + buf[(*b)++] = s >> 8; +} + +static inline void save_u32(u8 *buf, size_t *b, u32 u) +{ + buf[(*b)++] = u; + buf[(*b)++] = u >> 8; + buf[(*b)++] = u >> 16; + buf[(*b)++] = u >> 24; +} + +static inline u8 load_u8_(const u8 *buf, size_t *b) +{ + return buf[(*b)++]; +} + +static inline s8 load_s8_(const u8 *buf, size_t *b) +{ + return buf[(*b)++]; +} + +static inline u16 load_u16(const u8 *buf, size_t *b) +{ + u16 r = (buf[*b + 1] << 8) | buf[*b]; + (*b) += 2; + return r; +} + +static inline s16 load_s16(const u8 *buf, size_t *b) +{ + s16 r = (buf[*b + 1] << 8) | buf[*b]; + (*b) += 2; + return r; +} + +static inline u32 load_u32(const u8 *buf, size_t *b) +{ + u32 r = (buf[*b + 3] << 24) | (buf[*b + 2] << 16) | (buf[*b + 1] << 8) | buf[*b]; + (*b) += 4; + return r; +} -- 2.39.5