mirror of
				https://github.com/RaySollium99/picodrive.git
				synced 2025-10-26 16:29:37 -04:00 
			
		
		
		
	core, improve pico save/load (ADPCM state, page, etc)
This commit is contained in:
		
							parent
							
								
									f1b425e380
								
							
						
					
					
						commit
						fa4e0531d4
					
				
					 5 changed files with 136 additions and 78 deletions
				
			
		|  | @ -150,17 +150,14 @@ void PicoGetInternal(pint_t which, pint_ret_t *ret); | ||||||
| struct PicoEState; | struct PicoEState; | ||||||
| 
 | 
 | ||||||
| // pico.c
 | // pico.c
 | ||||||
| #define XPCM_BUFFER_SIZE (320+160) | #define XPCM_BUFFER_SIZE 64 | ||||||
| typedef struct | typedef struct | ||||||
| { | { | ||||||
| 	int pen_pos[2]; | 	int pen_pos[2]; | ||||||
| 	int page; | 	int page; | ||||||
| 	// internal
 |  | ||||||
| 	int fifo_bytes;      // bytes in FIFO
 | 	int fifo_bytes;      // bytes in FIFO
 | ||||||
| 	int fifo_bytes_prev; |  | ||||||
| 	int fifo_line_bytes; // float part, << 16
 |  | ||||||
| 	int line_counter; |  | ||||||
| 	unsigned short r1, r12; | 	unsigned short r1, r12; | ||||||
|  | 	unsigned int reserved[3]; | ||||||
| 	unsigned char xpcm_buffer[XPCM_BUFFER_SIZE+4]; | 	unsigned char xpcm_buffer[XPCM_BUFFER_SIZE+4]; | ||||||
| 	unsigned char *xpcm_ptr; | 	unsigned char *xpcm_ptr; | ||||||
| } picohw_state; | } picohw_state; | ||||||
|  |  | ||||||
|  | @ -18,6 +18,7 @@ picohw_state PicoPicohw; | ||||||
| PICO_INTERNAL void PicoReratePico(void) | PICO_INTERNAL void PicoReratePico(void) | ||||||
| { | { | ||||||
|   PicoPicoPCMRerate(); |   PicoPicoPCMRerate(); | ||||||
|  |   PicoPicohw.xpcm_ptr = PicoPicohw.xpcm_buffer + PicoPicohw.fifo_bytes; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| static void PicoLinePico(void) | static void PicoLinePico(void) | ||||||
|  |  | ||||||
							
								
								
									
										155
									
								
								pico/pico/xpcm.c
									
										
									
									
									
								
							
							
						
						
									
										155
									
								
								pico/pico/xpcm.c
									
										
									
									
									
								
							|  | @ -42,14 +42,23 @@ static const int step_deltas[16][16] = | ||||||
| 
 | 
 | ||||||
| static const int state_deltas[16] = { -1, -1, 0, 0, 1, 2, 2, 3, -1, -1, 0, 0, 1, 2, 2, 3 }; | static const int state_deltas[16] = { -1, -1, 0, 0, 1, 2, 2, 3, -1, -1, 0, 0, 1, 2, 2, 3 }; | ||||||
| 
 | 
 | ||||||
| static int sample = 0, state = 0; | s32 stepsamples = (44100LL<<16)/ADPCM_CLOCK; | ||||||
| static s32 stepsamples = (44100LL<<16)/ADPCM_CLOCK; |  | ||||||
| static s32 samplepos; |  | ||||||
| static int samplegain; |  | ||||||
| 
 | 
 | ||||||
| static int startpin, irqenable; | static struct xpcm_state { | ||||||
| static enum { RESET, START, HDR, COUNT } portstate = RESET; |   s32 samplepos;	// leftover duration for current sample wrt sndrate, Q16
 | ||||||
| static int rate, silence, nibbles, highlow, cache; |   int sample;		// current sample
 | ||||||
|  |   short state;		// ADPCM engine state
 | ||||||
|  |   short samplegain;	// programmable gain
 | ||||||
|  | 
 | ||||||
|  |   char startpin;	// value on the !START pin
 | ||||||
|  |   char irqenable;	// IRQ enabled?
 | ||||||
|  | 
 | ||||||
|  |   char portstate;	// data stream state
 | ||||||
|  |   short silence;	// silence blocks still to be played
 | ||||||
|  |   short rate, nibbles;	// ADPCM nibbles still to be played
 | ||||||
|  |   unsigned char highlow, cache;	// nibble selector and cache
 | ||||||
|  | } xpcm; | ||||||
|  | enum { RESET, START, HDR, COUNT }; // portstate
 | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
| // SEGA Pico specific filtering
 | // SEGA Pico specific filtering
 | ||||||
|  | @ -58,8 +67,8 @@ static int rate, silence, nibbles, highlow, cache; | ||||||
| #define FP(f)	(int)((f)*(1<<QB))      // convert to fixpoint
 | #define FP(f)	(int)((f)*(1<<QB))      // convert to fixpoint
 | ||||||
| 
 | 
 | ||||||
| static struct iir2 { // 2nd order IIR
 | static struct iir2 { // 2nd order IIR
 | ||||||
|   s32 a[2], gain; // coefficients
 |   s32 a[2], gain;	// coefficients
 | ||||||
|   s32 y[3], x[3]; // filter history
 |   s32 x[3], y[3];	// filter history
 | ||||||
| } filters[4]; | } filters[4]; | ||||||
| static struct iir2 *filter; | static struct iir2 *filter; | ||||||
| 
 | 
 | ||||||
|  | @ -85,6 +94,7 @@ static int PicoPicoFilterApply(struct iir2 *iir, int sample) | ||||||
|   if (!iir) |   if (!iir) | ||||||
|     return sample; |     return sample; | ||||||
| 
 | 
 | ||||||
|  |   // NB Butterworth specific!
 | ||||||
|   iir->x[0] = iir->x[1]; iir->x[1] = iir->x[2]; |   iir->x[0] = iir->x[1]; iir->x[1] = iir->x[2]; | ||||||
|   iir->x[2] = sample * iir->gain; // Qb
 |   iir->x[2] = sample * iir->gain; // Qb
 | ||||||
|   iir->y[0] = iir->y[1]; iir->y[1] = iir->y[2]; |   iir->y[0] = iir->y[1]; iir->y[1] = iir->y[2]; | ||||||
|  | @ -99,21 +109,21 @@ static int PicoPicoFilterApply(struct iir2 *iir, int sample) | ||||||
| PICO_INTERNAL void PicoPicoPCMResetN(int pin) | PICO_INTERNAL void PicoPicoPCMResetN(int pin) | ||||||
| { | { | ||||||
|   if (!pin) { |   if (!pin) { | ||||||
|     portstate = RESET; |     xpcm.portstate = RESET; | ||||||
|     sample = samplepos = state = 0; |     xpcm.sample = xpcm.samplepos = xpcm.state = 0; | ||||||
|     portstate = nibbles = silence = 0; |     xpcm.nibbles = xpcm.silence = 0; | ||||||
|   } else if (portstate == RESET) |   } else if (xpcm.portstate == RESET) | ||||||
|     portstate = START; |     xpcm.portstate = START; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| PICO_INTERNAL void PicoPicoPCMStartN(int pin) | PICO_INTERNAL void PicoPicoPCMStartN(int pin) | ||||||
| { | { | ||||||
|   startpin = pin; |   xpcm.startpin = pin; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| PICO_INTERNAL int PicoPicoPCMBusyN(void) | PICO_INTERNAL int PicoPicoPCMBusyN(void) | ||||||
| { | { | ||||||
|   return (portstate <= START); |   return (xpcm.portstate <= START); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
|  | @ -125,14 +135,14 @@ PICO_INTERNAL void PicoPicoPCMRerate(void) | ||||||
|   stepsamples = ((u64)PicoIn.sndRate<<16)/ADPCM_CLOCK; |   stepsamples = ((u64)PicoIn.sndRate<<16)/ADPCM_CLOCK; | ||||||
| 
 | 
 | ||||||
|   // compute filter coefficients, cutoff at half the ADPCM sample rate
 |   // compute filter coefficients, cutoff at half the ADPCM sample rate
 | ||||||
|   PicoPicoFilterCoeff(&filters[1],  5000/2, PicoIn.sndRate); // 5-6 KHz
 |   PicoPicoFilterCoeff(&filters[1],  6000/2, PicoIn.sndRate); // 5-6 KHz
 | ||||||
|   PicoPicoFilterCoeff(&filters[2],  8000/2, PicoIn.sndRate); // 8-12 KHz
 |   PicoPicoFilterCoeff(&filters[2],  9000/2, PicoIn.sndRate); // 8-12 KHz
 | ||||||
|   PicoPicoFilterCoeff(&filters[3], 14000/2, PicoIn.sndRate); // 14-16 KHz
 |   PicoPicoFilterCoeff(&filters[3], 15000/2, PicoIn.sndRate); // 14-16 KHz
 | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| PICO_INTERNAL void PicoPicoPCMGain(int gain) | PICO_INTERNAL void PicoPicoPCMGain(int gain) | ||||||
| { | { | ||||||
|   samplegain = gain*4; |   xpcm.samplegain = gain*4; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| PICO_INTERNAL void PicoPicoPCMFilter(int index) | PICO_INTERNAL void PicoPicoPCMFilter(int index) | ||||||
|  | @ -144,13 +154,14 @@ PICO_INTERNAL void PicoPicoPCMFilter(int index) | ||||||
| 
 | 
 | ||||||
| PICO_INTERNAL void PicoPicoPCMIrqEn(int enable) | PICO_INTERNAL void PicoPicoPCMIrqEn(int enable) | ||||||
| { | { | ||||||
|   irqenable = (enable ? 3 : 0); |   xpcm.irqenable = (enable ? 3 : 0); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| // TODO need an interupt pending mask?
 | // TODO need an interupt pending mask?
 | ||||||
| PICO_INTERNAL int PicoPicoIrqAck(int level) | PICO_INTERNAL int PicoPicoIrqAck(int level) | ||||||
| { | { | ||||||
|   return (PicoPicohw.fifo_bytes < FIFO_IRQ_THRESHOLD && level != irqenable ? irqenable : 0); |   return (PicoPicohw.fifo_bytes < FIFO_IRQ_THRESHOLD && level != xpcm.irqenable | ||||||
|  |             ? xpcm.irqenable : 0); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
|  | @ -158,18 +169,20 @@ PICO_INTERNAL int PicoPicoIrqAck(int level) | ||||||
| 
 | 
 | ||||||
| #define apply_filter(v) PicoPicoFilterApply(filter, v) | #define apply_filter(v) PicoPicoFilterApply(filter, v) | ||||||
| 
 | 
 | ||||||
|  | // compute next ADPCM sample
 | ||||||
| #define do_sample(nibble) \ | #define do_sample(nibble) \ | ||||||
| { \ | { \ | ||||||
|   sample += step_deltas[state][nibble]; \ |   xpcm.sample += step_deltas[xpcm.state][nibble]; \ | ||||||
|   state += state_deltas[nibble]; \ |   xpcm.state += state_deltas[nibble]; \ | ||||||
|   state = (state < 0 ? 0 : state > 15 ? 15 : state); \ |   xpcm.state = (xpcm.state < 0 ? 0 : xpcm.state > 15 ? 15 : xpcm.state); \ | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | // writes samples with sndRate, nearest neighbour resampling, filtering
 | ||||||
| #define write_sample(buffer, length, stereo) \ | #define write_sample(buffer, length, stereo) \ | ||||||
| { \ | { \ | ||||||
|   while (samplepos > 0 && length > 0) { \ |   while (xpcm.samplepos > 0 && length > 0) { \ | ||||||
|     int val = Limit(samplegain*sample, 16383, -16384); \ |     int val = Limit(xpcm.samplegain*xpcm.sample, 16383, -16384); \ | ||||||
|     samplepos -= 1<<16; \ |     xpcm.samplepos -= 1<<16; \ | ||||||
|     length --; \ |     length --; \ | ||||||
|     if (buffer) { \ |     if (buffer) { \ | ||||||
|       int out = apply_filter(val); \ |       int out = apply_filter(val); \ | ||||||
|  | @ -191,56 +204,56 @@ PICO_INTERNAL void PicoPicoPCMUpdate(short *buffer, int length, int stereo) | ||||||
|   // loop over FIFO data, generating ADPCM samples
 |   // loop over FIFO data, generating ADPCM samples
 | ||||||
|   while (length > 0 && src < lim) |   while (length > 0 && src < lim) | ||||||
|   { |   { | ||||||
|     if (silence > 0) { |     if (xpcm.silence > 0) { | ||||||
|       silence --; |       xpcm.silence --; | ||||||
|       sample = 0; |       xpcm.sample = 0; | ||||||
|       samplepos += stepsamples*256; |       xpcm.samplepos += stepsamples*256; | ||||||
| 
 | 
 | ||||||
|     } else if (nibbles > 0) { |     } else if (xpcm.nibbles > 0) { | ||||||
|       nibbles --; |       xpcm.nibbles --; | ||||||
| 
 | 
 | ||||||
|       if (highlow) |       if (xpcm.highlow) | ||||||
|         cache = *src++; |         xpcm.cache = *src++; | ||||||
|       else |       else | ||||||
|         cache <<= 4; |         xpcm.cache <<= 4; | ||||||
|       highlow = !highlow; |       xpcm.highlow = !xpcm.highlow; | ||||||
| 
 | 
 | ||||||
|       do_sample((cache & 0xf0) >> 4); |       do_sample((xpcm.cache & 0xf0) >> 4); | ||||||
|       samplepos += stepsamples*rate; |       xpcm.samplepos += stepsamples*xpcm.rate; | ||||||
| 
 | 
 | ||||||
|     } else switch (portstate) { |     } else switch (xpcm.portstate) { | ||||||
|       case RESET: |       case RESET: | ||||||
|         sample = 0; |         xpcm.sample = 0; | ||||||
|         samplepos += length<<16; |         xpcm.samplepos += length<<16; | ||||||
|         break; |         break; | ||||||
|       case START: |       case START: | ||||||
|         if (startpin) { |         if (xpcm.startpin) { | ||||||
|           if (*src) |           if (*src) | ||||||
|             portstate ++; |             xpcm.portstate ++; | ||||||
|           else // kill 0x00 bytes at stream start
 |           else // kill 0x00 bytes at stream start
 | ||||||
|             src ++; |             src ++; | ||||||
|         } else { |         } else { | ||||||
|           sample = 0; |           xpcm.sample = 0; | ||||||
|           samplepos += length<<16; |           xpcm.samplepos += length<<16; | ||||||
|         } |         } | ||||||
|         break; |         break; | ||||||
|       case HDR: |       case HDR: | ||||||
|         srcval = *src++; |         srcval = *src++; | ||||||
|         nibbles = silence = rate = 0; |         xpcm.nibbles = xpcm.silence = xpcm.rate = 0; | ||||||
|         highlow = 1; |         xpcm.highlow = 1; | ||||||
|         if (srcval == 0) { // terminator
 |         if (srcval == 0) { // terminator
 | ||||||
|           // HACK, kill leftover odd byte to avoid restart (Minna de Odorou)
 |           // HACK, kill leftover odd byte to avoid restart (Minna de Odorou)
 | ||||||
|           if (lim-src == 1) src++; |           if (lim-src == 1) src++; | ||||||
|           portstate = START; |           xpcm.portstate = START; | ||||||
|         } else switch (srcval >> 6) { |         } else switch (srcval >> 6) { | ||||||
|           case 0: silence = (srcval & 0x3f) + 1; break; |           case 0: xpcm.silence = (srcval & 0x3f) + 1; break; | ||||||
|           case 1: rate = (srcval & 0x3f) + 1; nibbles = 256; break; |           case 1: xpcm.rate = (srcval & 0x3f) + 1; xpcm.nibbles = 256; break; | ||||||
|           case 2: rate = (srcval & 0x3f) + 1; portstate = COUNT; break; |           case 2: xpcm.rate = (srcval & 0x3f) + 1; xpcm.portstate = COUNT; break; | ||||||
|           case 3: break; |           case 3: break; | ||||||
|         } |         } | ||||||
|         break; |         break; | ||||||
|       case COUNT: |       case COUNT: | ||||||
|         nibbles = *src++ + 1; portstate = HDR; |         xpcm.nibbles = *src++ + 1; xpcm.portstate = HDR; | ||||||
|         break; |         break; | ||||||
|       } |       } | ||||||
| 
 | 
 | ||||||
|  | @ -255,14 +268,14 @@ PICO_INTERNAL void PicoPicoPCMUpdate(short *buffer, int length, int stereo) | ||||||
|     elprintf(EL_PICOHW, "xpcm update: over %i", di); |     elprintf(EL_PICOHW, "xpcm update: over %i", di); | ||||||
| 
 | 
 | ||||||
|     if (!irq && di < FIFO_IRQ_THRESHOLD) |     if (!irq && di < FIFO_IRQ_THRESHOLD) | ||||||
|       irq = irqenable; |       irq = xpcm.irqenable; | ||||||
|     PicoPicohw.fifo_bytes = di; |     PicoPicohw.fifo_bytes = di; | ||||||
|   } else if (src == lim && src != PicoPicohw.xpcm_buffer) { |   } else if (src == lim && src != PicoPicohw.xpcm_buffer) { | ||||||
|     PicoPicohw.xpcm_ptr = PicoPicohw.xpcm_buffer; |     PicoPicohw.xpcm_ptr = PicoPicohw.xpcm_buffer; | ||||||
|     elprintf(EL_PICOHW, "xpcm update: under %i", length); |     elprintf(EL_PICOHW, "xpcm update: under %i", length); | ||||||
| 
 | 
 | ||||||
|     if (!irq) |     if (!irq) | ||||||
|       irq = irqenable; |       irq = xpcm.irqenable; | ||||||
|     PicoPicohw.fifo_bytes = 0; |     PicoPicohw.fifo_bytes = 0; | ||||||
|   } |   } | ||||||
| 
 | 
 | ||||||
|  | @ -275,7 +288,7 @@ PICO_INTERNAL void PicoPicoPCMUpdate(short *buffer, int length, int stereo) | ||||||
| 
 | 
 | ||||||
|   if (buffer && length) { |   if (buffer && length) { | ||||||
|     // for underflow, use last sample to avoid clicks
 |     // for underflow, use last sample to avoid clicks
 | ||||||
|     int val = Limit(samplegain*sample, 16383, -16384); |     int val = Limit(xpcm.samplegain*xpcm.sample, 16383, -16384); | ||||||
|     while (length--) { |     while (length--) { | ||||||
|       int out = apply_filter(val); |       int out = apply_filter(val); | ||||||
|       *buffer++ += out; |       *buffer++ += out; | ||||||
|  | @ -283,3 +296,31 @@ PICO_INTERNAL void PicoPicoPCMUpdate(short *buffer, int length, int stereo) | ||||||
|     } |     } | ||||||
|   } |   } | ||||||
| } | } | ||||||
|  | 
 | ||||||
|  | PICO_INTERNAL int PicoPicoPCMSave(void *buffer, int length) | ||||||
|  | { | ||||||
|  |   u8 *bp = buffer; | ||||||
|  | 
 | ||||||
|  |   if (length < sizeof(xpcm) + sizeof(filters)) { | ||||||
|  |     elprintf(EL_ANOMALY, "save buffer too small?"); | ||||||
|  |     return 0; | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   memcpy(bp, &xpcm, sizeof(xpcm)); | ||||||
|  |   bp += sizeof(xpcm); | ||||||
|  |   memcpy(bp, filters, sizeof(filters)); | ||||||
|  |   bp += sizeof(filters); | ||||||
|  |   return (bp - (u8*)buffer); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | PICO_INTERNAL void PicoPicoPCMLoad(void *buffer, int length) | ||||||
|  | { | ||||||
|  |   u8 *bp = buffer; | ||||||
|  | 
 | ||||||
|  |   if (length >= sizeof(xpcm)) | ||||||
|  |     memcpy(&xpcm, bp, sizeof(xpcm)); | ||||||
|  |   bp += sizeof(xpcm); | ||||||
|  |   if (length >= sizeof(xpcm) + sizeof(filters)) | ||||||
|  |     memcpy(filters, bp, sizeof(filters)); | ||||||
|  |   bp += sizeof(filters); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | @ -853,6 +853,8 @@ PICO_INTERNAL void PicoPicoPCMGain(int gain); | ||||||
| PICO_INTERNAL void PicoPicoPCMFilter(int index); | PICO_INTERNAL void PicoPicoPCMFilter(int index); | ||||||
| PICO_INTERNAL void PicoPicoPCMIrqEn(int enable); | PICO_INTERNAL void PicoPicoPCMIrqEn(int enable); | ||||||
| PICO_INTERNAL void PicoPicoPCMRerate(void); | PICO_INTERNAL void PicoPicoPCMRerate(void); | ||||||
|  | PICO_INTERNAL int PicoPicoPCMSave(void *buffer, int length); | ||||||
|  | PICO_INTERNAL void PicoPicoPCMLoad(void *buffer, int length); | ||||||
| 
 | 
 | ||||||
| // sek.c
 | // sek.c
 | ||||||
| PICO_INTERNAL void SekInit(void); | PICO_INTERNAL void SekInit(void); | ||||||
|  |  | ||||||
							
								
								
									
										49
									
								
								pico/state.c
									
										
									
									
									
								
							
							
						
						
									
										49
									
								
								pico/state.c
									
										
									
									
									
								
							|  | @ -133,6 +133,8 @@ typedef enum { | ||||||
|   CHUNK_CD_CDC, |   CHUNK_CD_CDC, | ||||||
|   CHUNK_CD_CDD, |   CHUNK_CD_CDD, | ||||||
|   CHUNK_YM2413, |   CHUNK_YM2413, | ||||||
|  |   CHUNK_PICO_PCM, | ||||||
|  |   CHUNK_PICO, | ||||||
|   //
 |   //
 | ||||||
|   CHUNK_DEFAULT_COUNT, |   CHUNK_DEFAULT_COUNT, | ||||||
|   CHUNK_CARTHW_ = CHUNK_CARTHW,  // 64 (defined in PicoInt)
 |   CHUNK_CARTHW_ = CHUNK_CARTHW,  // 64 (defined in PicoInt)
 | ||||||
|  | @ -181,9 +183,8 @@ static const char * const chunk_names[CHUNK_DEFAULT_COUNT] = { | ||||||
|   "SSH2 BIOS", // 35
 |   "SSH2 BIOS", // 35
 | ||||||
|   "SDRAM", |   "SDRAM", | ||||||
|   "DRAM", |   "DRAM", | ||||||
|   "PAL", |   "32X palette", | ||||||
|   "events", |   "32X events", | ||||||
|   "YM2413",   //40
 |  | ||||||
| }; | }; | ||||||
| 
 | 
 | ||||||
| static int write_chunk(chunk_name_e name, int len, void *data, void *file) | static int write_chunk(chunk_name_e name, int len, void *data, void *file) | ||||||
|  | @ -230,6 +231,10 @@ static int state_save(void *file) | ||||||
|   int retval = -1; |   int retval = -1; | ||||||
|   int len; |   int len; | ||||||
| 
 | 
 | ||||||
|  |   buf2 = malloc(CHUNK_LIMIT_W); | ||||||
|  |   if (buf2 == NULL) | ||||||
|  |     return -1; | ||||||
|  | 
 | ||||||
|   areaWrite("PicoSEXT", 1, 8, file); |   areaWrite("PicoSEXT", 1, 8, file); | ||||||
|   areaWrite(&ver, 1, 4, file); |   areaWrite(&ver, 1, 4, file); | ||||||
| 
 | 
 | ||||||
|  | @ -243,9 +248,15 @@ static int state_save(void *file) | ||||||
|     CHECKED_WRITE_BUFF(CHUNK_RAM,   PicoMem.ram); |     CHECKED_WRITE_BUFF(CHUNK_RAM,   PicoMem.ram); | ||||||
|     CHECKED_WRITE_BUFF(CHUNK_VSRAM, PicoMem.vsram); |     CHECKED_WRITE_BUFF(CHUNK_VSRAM, PicoMem.vsram); | ||||||
|     CHECKED_WRITE_BUFF(CHUNK_IOPORTS, PicoMem.ioports); |     CHECKED_WRITE_BUFF(CHUNK_IOPORTS, PicoMem.ioports); | ||||||
|     ym2612_pack_state(); |     if (PicoIn.AHW & PAHW_PICO) { | ||||||
|     ym_regs = YM2612GetRegs(); |       len = PicoPicoPCMSave(buf2, CHUNK_LIMIT_W); | ||||||
|     CHECKED_WRITE(CHUNK_FM, 0x200+4, ym_regs); |       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); | ||||||
|  |     } | ||||||
| 
 | 
 | ||||||
|     if (!(PicoIn.opt & POPT_DIS_IDLE_DET)) |     if (!(PicoIn.opt & POPT_DIS_IDLE_DET)) | ||||||
|       SekInitIdleDet(); |       SekInitIdleDet(); | ||||||
|  | @ -255,25 +266,23 @@ static int state_save(void *file) | ||||||
|     ym_regs = YM2413GetRegs(); |     ym_regs = YM2413GetRegs(); | ||||||
|     CHECKED_WRITE(CHUNK_YM2413, 0x40+4, ym_regs); |     CHECKED_WRITE(CHUNK_YM2413, 0x40+4, ym_regs); | ||||||
|   } |   } | ||||||
|  |   CHECKED_WRITE(CHUNK_PSG, 28*4, sn76496_regs); | ||||||
|  | 
 | ||||||
|  |   if (!(PicoIn.AHW & PAHW_PICO)) { | ||||||
|  |     z80_pack(buff_z80); | ||||||
|  |     CHECKED_WRITE_BUFF(CHUNK_Z80, buff_z80); | ||||||
|  |     CHECKED_WRITE_BUFF(CHUNK_ZRAM,  PicoMem.zram); | ||||||
|  |   } | ||||||
| 
 | 
 | ||||||
|   CHECKED_WRITE_BUFF(CHUNK_VRAM,  PicoMem.vram); |   CHECKED_WRITE_BUFF(CHUNK_VRAM,  PicoMem.vram); | ||||||
|   CHECKED_WRITE_BUFF(CHUNK_ZRAM,  PicoMem.zram); |  | ||||||
|   CHECKED_WRITE_BUFF(CHUNK_CRAM,  PicoMem.cram); |   CHECKED_WRITE_BUFF(CHUNK_CRAM,  PicoMem.cram); | ||||||
|   CHECKED_WRITE_BUFF(CHUNK_MISC,  Pico.m); |  | ||||||
| 
 | 
 | ||||||
|  |   CHECKED_WRITE_BUFF(CHUNK_MISC,  Pico.m); | ||||||
|   PicoVideoSave(); |   PicoVideoSave(); | ||||||
|   CHECKED_WRITE_BUFF(CHUNK_VIDEO, Pico.video); |   CHECKED_WRITE_BUFF(CHUNK_VIDEO, Pico.video); | ||||||
| 
 | 
 | ||||||
|   z80_pack(buff_z80); |  | ||||||
|   CHECKED_WRITE_BUFF(CHUNK_Z80, buff_z80); |  | ||||||
|   CHECKED_WRITE(CHUNK_PSG, 28*4, sn76496_regs); |  | ||||||
| 
 |  | ||||||
|   if (PicoIn.AHW & PAHW_MCD) |   if (PicoIn.AHW & PAHW_MCD) | ||||||
|   { |   { | ||||||
|     buf2 = malloc(CHUNK_LIMIT_W); |  | ||||||
|     if (buf2 == NULL) |  | ||||||
|       return -1; |  | ||||||
| 
 |  | ||||||
|     memset(buff, 0, sizeof(buff)); |     memset(buff, 0, sizeof(buff)); | ||||||
|     SekPackCpu(buff, 1); |     SekPackCpu(buff, 1); | ||||||
|     if (Pico_mcd->s68k_regs[3] & 4) // 1M mode?
 |     if (Pico_mcd->s68k_regs[3] & 4) // 1M mode?
 | ||||||
|  | @ -464,6 +473,14 @@ static int state_load(void *file) | ||||||
|         ym2612_unpack_state(); |         ym2612_unpack_state(); | ||||||
|         break; |         break; | ||||||
| 
 | 
 | ||||||
|  |       case CHUNK_PICO_PCM: | ||||||
|  |         CHECKED_READ(len, buf); | ||||||
|  |         PicoPicoPCMLoad(buf, len); | ||||||
|  |         break; | ||||||
|  |       case CHUNK_PICO: | ||||||
|  |         CHECKED_READ_BUFF(PicoPicohw); | ||||||
|  |         break; | ||||||
|  | 
 | ||||||
|       case CHUNK_SMS: |       case CHUNK_SMS: | ||||||
|         CHECKED_READ_BUFF(Pico.ms); |         CHECKED_READ_BUFF(Pico.ms); | ||||||
|         break; |         break; | ||||||
|  |  | ||||||
		Loading…
	
	Add table
		Add a link
		
	
		Reference in a new issue
	
	 kub
						kub