Mercurial > repos > blastem
comparison gst.c @ 487:c08a4efeee7f opengl
Update opengl branch from default. Fix build breakage unrelated to merge
author | Mike Pavone <pavone@retrodev.com> |
---|---|
date | Sat, 26 Oct 2013 22:38:47 -0700 |
parents | 140af5509ce7 |
children | 5439ae7946ca |
comparison
equal
deleted
inserted
replaced
449:7696d824489d | 487:c08a4efeee7f |
---|---|
1 /* | |
2 Copyright 2013 Michael Pavone | |
3 This file is part of BlastEm. | |
4 BlastEm is free software distributed under the terms of the GNU General Public License version 3 or greater. See COPYING for full license text. | |
5 */ | |
6 #include "gst.h" | |
7 #include <string.h> | |
8 | |
9 #define GST_68K_REGS 0x80 | |
10 #define GST_68K_REG_SIZE (0xDA-GST_68K_REGS) | |
11 #define GST_68K_PC_OFFSET (0xC8-GST_68K_REGS) | |
12 #define GST_68K_SR_OFFSET (0xD0-GST_68K_REGS) | |
13 #define GST_68K_USP_OFFSET (0xD2-GST_68K_REGS) | |
14 #define GST_68K_SSP_OFFSET (0xD6-GST_68K_REGS) | |
15 #define GST_68K_RAM 0x2478 | |
16 #define GST_Z80_REGS 0x404 | |
17 #define GST_Z80_REG_SIZE (0x440-GST_Z80_REGS) | |
18 #define GST_Z80_RAM 0x474 | |
19 #define GST_VDP_REGS 0xFA | |
20 #define GST_VDP_MEM 0x12478 | |
21 #define GST_YM_OFFSET 0x1E4 | |
22 #define GST_YM_SIZE (0x3E4-GST_YM_OFFSET) | |
23 | |
24 uint32_t read_le_32(uint8_t * data) | |
25 { | |
26 return data[3] << 24 | data[2] << 16 | data[1] << 8 | data[0]; | |
27 } | |
28 | |
29 uint16_t read_le_16(uint8_t * data) | |
30 { | |
31 return data[1] << 8 | data[0]; | |
32 } | |
33 | |
34 uint16_t read_be_16(uint8_t * data) | |
35 { | |
36 return data[0] << 8 | data[1]; | |
37 } | |
38 | |
39 void write_le_32(uint8_t * dst, uint32_t val) | |
40 { | |
41 dst[0] = val; | |
42 dst[1] = val >> 8; | |
43 dst[2] = val >> 16; | |
44 dst[3] = val >> 24; | |
45 } | |
46 | |
47 void write_le_16(uint8_t * dst, uint16_t val) | |
48 { | |
49 dst[0] = val; | |
50 dst[1] = val >> 8; | |
51 } | |
52 | |
53 void write_be_32(uint8_t * dst, uint32_t val) | |
54 { | |
55 dst[0] = val >> 24; | |
56 dst[1] = val >> 16; | |
57 dst[2] = val >> 8; | |
58 dst[3] = val; | |
59 } | |
60 | |
61 void write_be_16(uint8_t * dst, uint16_t val) | |
62 { | |
63 dst[0] = val >> 8; | |
64 dst[1] = val; | |
65 } | |
66 | |
67 uint32_t m68k_load_gst(m68k_context * context, FILE * gstfile) | |
68 { | |
69 uint8_t buffer[4096]; | |
70 fseek(gstfile, GST_68K_REGS, SEEK_SET); | |
71 if (fread(buffer, 1, GST_68K_REG_SIZE, gstfile) != GST_68K_REG_SIZE) { | |
72 fputs("Failed to read 68K registers from savestate\n", stderr); | |
73 return 0; | |
74 } | |
75 uint8_t * curpos = buffer; | |
76 for (int i = 0; i < 8; i++) { | |
77 context->dregs[i] = read_le_32(curpos); | |
78 curpos += sizeof(uint32_t); | |
79 } | |
80 for (int i = 0; i < 8; i++) { | |
81 context->aregs[i] = read_le_32(curpos); | |
82 curpos += sizeof(uint32_t); | |
83 } | |
84 uint32_t pc = read_le_32(buffer + GST_68K_PC_OFFSET); | |
85 uint16_t sr = read_le_16(buffer + GST_68K_SR_OFFSET); | |
86 context->status = sr >> 8; | |
87 for (int flag = 4; flag >= 0; flag--) { | |
88 context->flags[flag] = sr & 1; | |
89 sr >>= 1; | |
90 } | |
91 if (context->status & (1 << 5)) { | |
92 context->aregs[8] = read_le_32(buffer + GST_68K_USP_OFFSET); | |
93 } else { | |
94 context->aregs[8] = read_le_32(buffer + GST_68K_SSP_OFFSET); | |
95 } | |
96 fseek(gstfile, GST_68K_RAM, SEEK_SET); | |
97 for (int i = 0; i < (32*1024);) { | |
98 if (fread(buffer, 1, sizeof(buffer), gstfile) != sizeof(buffer)) { | |
99 fputs("Failed to read 68K RAM from savestate\n", stderr); | |
100 return 0; | |
101 } | |
102 for(curpos = buffer; curpos < (buffer + sizeof(buffer)); curpos += sizeof(uint16_t)) { | |
103 context->mem_pointers[1][i++] = read_be_16(curpos); | |
104 } | |
105 } | |
106 return pc; | |
107 } | |
108 | |
109 uint8_t m68k_save_gst(m68k_context * context, uint32_t pc, FILE * gstfile) | |
110 { | |
111 uint8_t buffer[4096]; | |
112 uint8_t * curpos = buffer; | |
113 for (int i = 0; i < 8; i++) { | |
114 write_le_32(curpos, context->dregs[i]); | |
115 curpos += sizeof(uint32_t); | |
116 } | |
117 for (int i = 0; i < 8; i++) { | |
118 write_le_32(curpos, context->aregs[i]); | |
119 curpos += sizeof(uint32_t); | |
120 } | |
121 write_le_32(buffer + GST_68K_PC_OFFSET, pc); | |
122 uint16_t sr = context->status << 3; | |
123 for (int flag = 4; flag >= 0; flag--) { | |
124 sr <<= 1; | |
125 sr |= context->flags[flag]; | |
126 } | |
127 write_le_16(buffer + GST_68K_SR_OFFSET, sr); | |
128 if (context->status & (1 << 5)) { | |
129 write_le_32(buffer + GST_68K_USP_OFFSET, context->aregs[8]); | |
130 write_le_32(buffer + GST_68K_SSP_OFFSET, context->aregs[7]); | |
131 } else { | |
132 write_le_32(buffer + GST_68K_USP_OFFSET, context->aregs[7]); | |
133 write_le_32(buffer + GST_68K_SSP_OFFSET, context->aregs[8]); | |
134 } | |
135 fseek(gstfile, GST_68K_REGS, SEEK_SET); | |
136 if (fwrite(buffer, 1, GST_68K_REG_SIZE, gstfile) != GST_68K_REG_SIZE) { | |
137 fputs("Failed to write 68K registers to savestate\n", stderr); | |
138 return 0; | |
139 } | |
140 | |
141 fseek(gstfile, GST_68K_RAM, SEEK_SET); | |
142 for (int i = 0; i < (32*1024);) { | |
143 for(curpos = buffer; curpos < (buffer + sizeof(buffer)); curpos += sizeof(uint16_t)) { | |
144 write_be_16(curpos, context->mem_pointers[1][i++]); | |
145 } | |
146 if (fwrite(buffer, 1, sizeof(buffer), gstfile) != sizeof(buffer)) { | |
147 fputs("Failed to write 68K RAM to savestate\n", stderr); | |
148 return 0; | |
149 } | |
150 } | |
151 return 1; | |
152 } | |
153 | |
154 uint8_t z80_load_gst(z80_context * context, FILE * gstfile) | |
155 { | |
156 uint8_t regdata[GST_Z80_REG_SIZE]; | |
157 fseek(gstfile, GST_Z80_REGS, SEEK_SET); | |
158 if (fread(regdata, 1, sizeof(regdata), gstfile) != sizeof(regdata)) { | |
159 fputs("Failed to read Z80 registers from savestate\n", stderr); | |
160 return 0; | |
161 } | |
162 uint8_t * curpos = regdata; | |
163 uint8_t f = *(curpos++); | |
164 context->flags[ZF_C] = f & 1; | |
165 f >>= 1; | |
166 context->flags[ZF_N] = f & 1; | |
167 f >>= 1; | |
168 context->flags[ZF_PV] = f & 1; | |
169 f >>= 2; | |
170 context->flags[ZF_H] = f & 1; | |
171 f >>= 2; | |
172 context->flags[ZF_Z] = f & 1; | |
173 f >>= 1; | |
174 context->flags[ZF_S] = f; | |
175 | |
176 context->regs[Z80_A] = *curpos; | |
177 curpos += 3; | |
178 for (int reg = Z80_C; reg <= Z80_IYH; reg++) { | |
179 context->regs[reg++] = *(curpos++); | |
180 context->regs[reg] = *curpos; | |
181 curpos += 3; | |
182 } | |
183 context->pc = read_le_16(curpos); | |
184 curpos += 4; | |
185 context->sp = read_le_16(curpos); | |
186 curpos += 4; | |
187 f = *(curpos++); | |
188 context->alt_flags[ZF_C] = f & 1; | |
189 f >>= 1; | |
190 context->alt_flags[ZF_N] = f & 1; | |
191 f >>= 1; | |
192 context->alt_flags[ZF_PV] = f & 1; | |
193 f >>= 2; | |
194 context->alt_flags[ZF_H] = f & 1; | |
195 f >>= 2; | |
196 context->alt_flags[ZF_Z] = f & 1; | |
197 f >>= 1; | |
198 context->alt_flags[ZF_S] = f; | |
199 context->alt_regs[Z80_A] = *curpos; | |
200 curpos += 3; | |
201 for (int reg = Z80_C; reg <= Z80_H; reg++) { | |
202 context->alt_regs[reg++] = *(curpos++); | |
203 context->alt_regs[reg] = *curpos; | |
204 curpos += 3; | |
205 } | |
206 context->regs[Z80_I] = *curpos; | |
207 curpos += 2; | |
208 context->iff1 = context->iff2 = *curpos; | |
209 curpos += 2; | |
210 reset = !*(curpos++); | |
211 busreq = *curpos; | |
212 curpos += 3; | |
213 uint32_t bank = read_le_32(curpos); | |
214 if (bank < 0x400000) { | |
215 context->mem_pointers[1] = context->mem_pointers[2] + bank; | |
216 } else { | |
217 context->mem_pointers[1] = NULL; | |
218 } | |
219 context->bank_reg = bank >> 15; | |
220 fseek(gstfile, GST_Z80_RAM, SEEK_SET); | |
221 if(fread(context->mem_pointers[0], 1, 8*1024, gstfile) != (8*1024)) { | |
222 fputs("Failed to read Z80 RAM from savestate\n", stderr); | |
223 return 0; | |
224 } | |
225 return 1; | |
226 } | |
227 | |
228 uint8_t vdp_load_gst(vdp_context * context, FILE * state_file) | |
229 { | |
230 uint8_t tmp_buf[CRAM_SIZE*2]; | |
231 fseek(state_file, GST_VDP_REGS, SEEK_SET); | |
232 if (fread(context->regs, 1, VDP_REGS, state_file) != VDP_REGS) { | |
233 fputs("Failed to read VDP registers from savestate\n", stderr); | |
234 return 0; | |
235 } | |
236 context->double_res = (context->regs[REG_MODE_4] & (BIT_INTERLACE | BIT_DOUBLE_RES)) == (BIT_INTERLACE | BIT_DOUBLE_RES); | |
237 if (!context->double_res) { | |
238 context->framebuf = context->oddbuf; | |
239 } | |
240 latch_mode(context); | |
241 if (fread(tmp_buf, 1, sizeof(tmp_buf), state_file) != sizeof(tmp_buf)) { | |
242 fputs("Failed to read CRAM from savestate\n", stderr); | |
243 return 0; | |
244 } | |
245 for (int i = 0; i < CRAM_SIZE; i++) { | |
246 uint16_t value; | |
247 context->cram[i] = value = (tmp_buf[i*2+1] << 8) | tmp_buf[i*2]; | |
248 context->colors[i] = color_map[value & 0xEEE]; | |
249 context->colors[i + CRAM_SIZE] = color_map[(value & 0xEEE) | FBUF_SHADOW]; | |
250 context->colors[i + CRAM_SIZE*2] = color_map[(value & 0xEEE) | FBUF_HILIGHT]; | |
251 } | |
252 if (fread(tmp_buf, 2, VSRAM_SIZE, state_file) != VSRAM_SIZE) { | |
253 fputs("Failed to read VSRAM from savestate\n", stderr); | |
254 return 0; | |
255 } | |
256 for (int i = 0; i < VSRAM_SIZE; i++) { | |
257 context->vsram[i] = (tmp_buf[i*2+1] << 8) | tmp_buf[i*2]; | |
258 } | |
259 fseek(state_file, GST_VDP_MEM, SEEK_SET); | |
260 if (fread(context->vdpmem, 1, VRAM_SIZE, state_file) != VRAM_SIZE) { | |
261 fputs("Failed to read VRAM from savestate\n", stderr); | |
262 return 0; | |
263 } | |
264 return 1; | |
265 } | |
266 | |
267 uint8_t vdp_save_gst(vdp_context * context, FILE * outfile) | |
268 { | |
269 uint8_t tmp_buf[CRAM_SIZE*2]; | |
270 fseek(outfile, GST_VDP_REGS, SEEK_SET); | |
271 if(fwrite(context->regs, 1, VDP_REGS, outfile) != VDP_REGS) { | |
272 fputs("Error writing VDP regs to savestate\n", stderr); | |
273 return 0; | |
274 } | |
275 for (int i = 0; i < CRAM_SIZE; i++) | |
276 { | |
277 tmp_buf[i*2] = context->cram[i]; | |
278 tmp_buf[i*2+1] = context->cram[i] >> 8; | |
279 } | |
280 if (fwrite(tmp_buf, 1, sizeof(tmp_buf), outfile) != sizeof(tmp_buf)) { | |
281 fputs("Error writing CRAM to savestate\n", stderr); | |
282 return 0; | |
283 } | |
284 for (int i = 0; i < VSRAM_SIZE; i++) | |
285 { | |
286 tmp_buf[i*2] = context->vsram[i]; | |
287 tmp_buf[i*2+1] = context->vsram[i] >> 8; | |
288 } | |
289 if (fwrite(tmp_buf, 2, VSRAM_SIZE, outfile) != VSRAM_SIZE) { | |
290 fputs("Error writing VSRAM to savestate\n", stderr); | |
291 return 0; | |
292 } | |
293 fseek(outfile, GST_VDP_MEM, SEEK_SET); | |
294 if (fwrite(context->vdpmem, 1, VRAM_SIZE, outfile) != VRAM_SIZE) { | |
295 fputs("Error writing VRAM to savestate\n", stderr); | |
296 return 0; | |
297 } | |
298 return 1; | |
299 } | |
300 | |
301 uint8_t z80_save_gst(z80_context * context, FILE * gstfile) | |
302 { | |
303 uint8_t regdata[GST_Z80_REG_SIZE]; | |
304 uint8_t * curpos = regdata; | |
305 memset(regdata, 0, sizeof(regdata)); | |
306 uint8_t f = context->flags[ZF_S]; | |
307 f <<= 1; | |
308 f |= context->flags[ZF_Z] ; | |
309 f <<= 2; | |
310 f |= context->flags[ZF_H]; | |
311 f <<= 2; | |
312 f |= context->flags[ZF_PV]; | |
313 f <<= 1; | |
314 f |= context->flags[ZF_N]; | |
315 f <<= 1; | |
316 f |= context->flags[ZF_C]; | |
317 *(curpos++) = f; | |
318 *curpos = context->regs[Z80_A]; | |
319 | |
320 curpos += 3; | |
321 for (int reg = Z80_C; reg <= Z80_IYH; reg++) { | |
322 *(curpos++) = context->regs[reg++]; | |
323 *curpos = context->regs[reg]; | |
324 curpos += 3; | |
325 } | |
326 write_le_16(curpos, context->pc); | |
327 curpos += 4; | |
328 write_le_16(curpos, context->sp); | |
329 curpos += 4; | |
330 f = context->alt_flags[ZF_S]; | |
331 f <<= 1; | |
332 f |= context->alt_flags[ZF_Z] ; | |
333 f <<= 2; | |
334 f |= context->alt_flags[ZF_H]; | |
335 f <<= 2; | |
336 f |= context->alt_flags[ZF_PV]; | |
337 f <<= 1; | |
338 f |= context->alt_flags[ZF_N]; | |
339 f <<= 1; | |
340 f |= context->alt_flags[ZF_C]; | |
341 *(curpos++) = f; | |
342 *curpos = context->alt_regs[Z80_A]; | |
343 curpos += 3; | |
344 for (int reg = Z80_C; reg <= Z80_H; reg++) { | |
345 *(curpos++) = context->alt_regs[reg++]; | |
346 *curpos = context->alt_regs[reg]; | |
347 curpos += 3; | |
348 } | |
349 *curpos = context->regs[Z80_I]; | |
350 curpos += 2; | |
351 *curpos = context->iff1; | |
352 curpos += 2; | |
353 *(curpos++) = !reset; | |
354 *curpos = busreq; | |
355 curpos += 3; | |
356 uint32_t bank = context->bank_reg << 15; | |
357 write_le_32(curpos, bank); | |
358 fseek(gstfile, GST_Z80_REGS, SEEK_SET); | |
359 if (fwrite(regdata, 1, sizeof(regdata), gstfile) != sizeof(regdata)) { | |
360 return 0; | |
361 } | |
362 fseek(gstfile, GST_Z80_RAM, SEEK_SET); | |
363 if(fwrite(context->mem_pointers[0], 1, 8*1024, gstfile) != (8*1024)) { | |
364 fputs("Failed to write Z80 RAM to savestate\n", stderr); | |
365 return 0; | |
366 } | |
367 return 1; | |
368 } | |
369 | |
370 uint8_t ym_load_gst(ym2612_context * context, FILE * gstfile) | |
371 { | |
372 uint8_t regdata[GST_YM_SIZE]; | |
373 fseek(gstfile, GST_YM_OFFSET, SEEK_SET); | |
374 if (fread(regdata, 1, sizeof(regdata), gstfile) != sizeof(regdata)) { | |
375 return 0; | |
376 } | |
377 for (int i = 0; i < sizeof(regdata); i++) { | |
378 if (i & 0x100) { | |
379 ym_address_write_part2(context, i & 0xFF); | |
380 } else { | |
381 ym_address_write_part1(context, i); | |
382 } | |
383 ym_data_write(context, regdata[i]); | |
384 } | |
385 return 1; | |
386 } | |
387 | |
388 uint8_t ym_save_gst(ym2612_context * context, FILE * gstfile) | |
389 { | |
390 uint8_t regdata[GST_YM_SIZE]; | |
391 for (int i = 0; i < sizeof(regdata); i++) { | |
392 if (i & 0x100) { | |
393 int reg = (i & 0xFF); | |
394 if (reg >= YM_PART2_START && reg < YM_REG_END) { | |
395 regdata[i] = context->part2_regs[reg-YM_PART2_START]; | |
396 } else { | |
397 regdata[i] = 0xFF; | |
398 } | |
399 } else { | |
400 if (i >= YM_PART1_START && i < YM_REG_END) { | |
401 regdata[i] = context->part1_regs[i-YM_PART1_START]; | |
402 } else { | |
403 regdata[i] = 0xFF; | |
404 } | |
405 } | |
406 } | |
407 fseek(gstfile, GST_YM_OFFSET, SEEK_SET); | |
408 if (fwrite(regdata, 1, sizeof(regdata), gstfile) != sizeof(regdata)) { | |
409 return 0; | |
410 } | |
411 return 1; | |
412 } | |
413 | |
414 uint32_t load_gst(genesis_context * gen, char * fname) | |
415 { | |
416 FILE * gstfile = fopen(fname, "rb"); | |
417 if (!gstfile) { | |
418 fprintf(stderr, "Could not open file %s for reading\n", fname); | |
419 goto error; | |
420 } | |
421 char ident[5]; | |
422 if (fread(ident, 1, sizeof(ident), gstfile) != sizeof(ident)) { | |
423 fprintf(stderr, "Could not read ident code from %s\n", fname); | |
424 goto error_close; | |
425 } | |
426 if (memcmp(ident, "GST\x40\xE0", 5) != 0) { | |
427 fprintf(stderr, "%s doesn't appear to be a GST savestate. The ident code is %c%c%c\\x%X\\x%X instead of GST\\x40\\xE0.\n", fname, ident[0], ident[1], ident[2], ident[3], ident[4]); | |
428 goto error_close; | |
429 } | |
430 uint32_t pc = m68k_load_gst(gen->m68k, gstfile); | |
431 if (!pc) { | |
432 goto error_close; | |
433 } | |
434 if (!vdp_load_gst(gen->vdp, gstfile)) { | |
435 goto error_close; | |
436 } | |
437 if (!ym_load_gst(gen->ym, gstfile)) { | |
438 goto error_close; | |
439 } | |
440 if (!z80_load_gst(gen->z80, gstfile)) { | |
441 goto error_close; | |
442 } | |
443 gen->ports[0].control = 0x40; | |
444 gen->ports[1].control = 0x40; | |
445 fclose(gstfile); | |
446 return pc; | |
447 | |
448 error_close: | |
449 fclose(gstfile); | |
450 error: | |
451 return 0; | |
452 } | |
453 | |
454 uint8_t save_gst(genesis_context * gen, char *fname, uint32_t m68k_pc) | |
455 { | |
456 FILE * gstfile = fopen(fname, "wb"); | |
457 if (!gstfile) { | |
458 fprintf(stderr, "Could not open %s for writing\n", fname); | |
459 goto error; | |
460 } | |
461 if (fwrite("GST\x40\xE0", 1, 5, gstfile) != 5) { | |
462 fputs("Error writing signature to savestate\n", stderr); | |
463 goto error_close; | |
464 } | |
465 if (!m68k_save_gst(gen->m68k, m68k_pc, gstfile)) { | |
466 goto error_close; | |
467 } | |
468 if (!z80_save_gst(gen->z80, gstfile)) { | |
469 goto error_close; | |
470 } | |
471 if (!vdp_save_gst(gen->vdp, gstfile)) { | |
472 goto error_close; | |
473 } | |
474 if (!ym_save_gst(gen->ym, gstfile)) { | |
475 goto error_close; | |
476 } | |
477 return 1; | |
478 | |
479 error_close: | |
480 fclose(gstfile); | |
481 error: | |
482 return 0; | |
483 } |