Merge pull request #3 from hiroshica/hwork

This commit is contained in:
kub 2020-07-28 23:29:24 +02:00
commit 3df6254bb4
22 changed files with 348 additions and 66 deletions

4
.gitignore vendored
View file

@ -19,3 +19,7 @@ obj/
.opk_data .opk_data
PicoDrive PicoDrive
PicoDrive.opk PicoDrive.opk
pico_int_offs.h
amalgamate
textfilter

3
.gitmodules vendored
View file

@ -4,3 +4,6 @@
[submodule "cpu/cyclone"] [submodule "cpu/cyclone"]
path = cpu/cyclone path = cpu/cyclone
url = https://github.com/notaz/cyclone68000.git url = https://github.com/notaz/cyclone68000.git
[submodule "pico/sound/emu2413"]
path = pico/sound/emu2413
url = https://github.com/digital-sound-antiques/emu2413.git

View file

@ -53,4 +53,5 @@ Additional thanks
* Paul Cercueil for OpenDingux port. * Paul Cercueil for OpenDingux port.
* Inder for some graphics. * Inder for some graphics.
* squarepusher for some libretro fixes * squarepusher for some libretro fixes
* Hiroshica for support of japanese Mark-III extended YM2413 sound
* Anyone else I forgot. Let me know if it's you. * Anyone else I forgot. Let me know if it's you.

View file

@ -34,6 +34,7 @@ gp2x,wiz,caanoo|open2x with ubuntu arm gcc 4.7|CROSS_COMPILE=arm-linux-gnueabi-
opendingux|opendingux|CROSS_COMPILE=mipsel-linux- CFLAGS="-I$TC/usr/include -I$TC/usr/include/SDL" LDFLAGS="--sysroot $TC -L$TC/lib" ./configure --platform=opendingux opendingux|opendingux|CROSS_COMPILE=mipsel-linux- CFLAGS="-I$TC/usr/include -I$TC/usr/include/SDL" LDFLAGS="--sysroot $TC -L$TC/lib" ./configure --platform=opendingux
opendingux|opendingux with ubuntu mips gcc 5.4|CROSS_COMPILE=mipsel-linux-gnu- CFLAGS="-I$TC/usr/include -I$TC/usr/include/SDL" LDFLAGS="-B$TC/usr/lib -B$TC/lib -Wl,-rpath-link=$TC/usr/lib -Wl,-rpath-link=$TC/lib" ./configure --platform=opendingux opendingux|opendingux with ubuntu mips gcc 5.4|CROSS_COMPILE=mipsel-linux-gnu- CFLAGS="-I$TC/usr/include -I$TC/usr/include/SDL" LDFLAGS="-B$TC/usr/lib -B$TC/lib -Wl,-rpath-link=$TC/usr/lib -Wl,-rpath-link=$TC/lib" ./configure --platform=opendingux
gcw0|gcw0|CROSS_COMPILE=mipsel-gcw0-linux-uclibc- CFLAGS="-I$TC/usr/mipsel-gcw0-linux-uclibc/sysroot/usr/include -I$TC/usr/mipsel-gcw0-linux-uclibc/sysroot/usr/include/SDL" LDFLAGS="--sysroot $TC/usr/mipsel-gcw0-linux-uclibc/sysroot" ./configure --platform=gcw0 gcw0|gcw0|CROSS_COMPILE=mipsel-gcw0-linux-uclibc- CFLAGS="-I$TC/usr/mipsel-gcw0-linux-uclibc/sysroot/usr/include -I$TC/usr/mipsel-gcw0-linux-uclibc/sysroot/usr/include/SDL" LDFLAGS="--sysroot $TC/usr/mipsel-gcw0-linux-uclibc/sysroot" ./configure --platform=gcw0
rg350|rg350|CROSS_COMPILE=mipsel-linux- CFLAGS="-I$TC/usr/include -I$TC/usr/include/SDL" LDFLAGS="--sysroot $TC -L$TC/lib" ./configure --platform=rg350
For gp2x, wiz, and caanoo you may need to compile libpng first. For gp2x, wiz, and caanoo you may need to compile libpng first.

15
configure vendored
View file

@ -38,7 +38,7 @@ check_define()
# setting options to "yes" or "no" will make that choice default, # setting options to "yes" or "no" will make that choice default,
# "" means "autodetect". # "" means "autodetect".
platform_list="generic pandora gp2x wiz caanoo opendingux gcw0 rpi1 rpi2" platform_list="generic pandora gp2x wiz caanoo opendingux gcw0 rg350 rpi1 rpi2"
platform="generic" platform="generic"
sound_driver_list="oss alsa sdl" sound_driver_list="oss alsa sdl"
sound_drivers="" sound_drivers=""
@ -62,7 +62,12 @@ CC="${CC-${CROSS_COMPILE}gcc}"
CXX="${CXX-${CROSS_COMPILE}g++}" CXX="${CXX-${CROSS_COMPILE}g++}"
AS="${AS-${CROSS_COMPILE}as}" AS="${AS-${CROSS_COMPILE}as}"
STRIP="${STRIP-${CROSS_COMPILE}strip}" STRIP="${STRIP-${CROSS_COMPILE}strip}"
test -n "$SDL_CONFIG" || SDL_CONFIG="`$CC $CFLAGS $LDFLAGS --print-sysroot 2> /dev/null || true`/usr/bin/sdl-config" SYSROOT=`$CC $CFLAGS $LDFLAGS --print-sysroot 2> /dev/null || true`
test -n "$SDL_CONFIG" || SDL_CONFIG="$(ls $SYSROOT/*bin*/sdl-config 2>/dev/null | grep /bin/sdl-config | head -n 1)"
test -n "$SDL_CONFIG" || SDL_CONFIG="$(ls $SYSROOT/*/*bin*/sdl-config 2>/dev/null | grep /bin/sdl-config | head -n 1)"
#test -n "$SDL_CONFIG" || SDL_CONFIG="$(ls $SYSROOT/*bin*/sdl2-config 2>/dev/null | grep /bin/sdl2-config | head -n 1)"
#test -n "$SDL_CONFIG" || SDL_CONFIG="$(ls $SYSROOT/*/*bin*/sdl2-config 2>/dev/null | grep /bin/sdl2-config | head -n 1)"
SDLVERSION=sdl && echo $SDL_CONFIG | grep -q sdl2 && SDLVERSION=sdl2
MAIN_LDLIBS="$LDLIBS -lm" MAIN_LDLIBS="$LDLIBS -lm"
config_mak="config.mak" config_mak="config.mak"
@ -86,9 +91,10 @@ set_platform()
;; ;;
generic) generic)
;; ;;
opendingux | gcw0) opendingux | gcw0 | rg350)
sound_drivers="sdl" sound_drivers="sdl"
# both are really an opendingux # both are really an opendingux
CFLAGS="$CFLAGS -D__`echo $platform | tr '[a-z]' '[A-Z]'`__"
platform="opendingux" platform="opendingux"
;; ;;
pandora) pandora)
@ -376,6 +382,9 @@ if [ "$need_sdl" = "yes" ]; then
CFLAGS="$CFLAGS `$SDL_CONFIG --cflags`" CFLAGS="$CFLAGS `$SDL_CONFIG --cflags`"
MAIN_LDLIBS="`$SDL_CONFIG --libs` $MAIN_LDLIBS" MAIN_LDLIBS="`$SDL_CONFIG --libs` $MAIN_LDLIBS"
check_sdl `$SDL_CONFIG --libs` || fail "please install libsdl (libsdl1.2-dev)" check_sdl `$SDL_CONFIG --libs` || fail "please install libsdl (libsdl1.2-dev)"
if [ "$SDLVERSION" = "sdl2" ]; then
CFLAGS="$CFLAGS -D__USE_SDL2__"
fi
fi fi
if check_option -Wno-unused_result; then if check_option -Wno-unused_result; then

View file

@ -19,6 +19,41 @@ static void (*FinalizeLineM4)(int line);
static int skip_next_line; static int skip_next_line;
static int screen_offset; static int screen_offset;
#define PLANAR_PIXELL(x,p) \
t = pack & (0x80808080 >> p); \
t = ((t >> (7-p)) | (t >> (14-p)) | (t >> (21-p)) | (t >> (28-p))) & 0x0f; \
pd[x] = pal|t;
static void TileNormM4Low(int sx, unsigned int pack, int pal)
{
unsigned char *pd = Pico.est.HighCol + sx;
unsigned int t;
PLANAR_PIXELL(0, 0)
PLANAR_PIXELL(1, 1)
PLANAR_PIXELL(2, 2)
PLANAR_PIXELL(3, 3)
PLANAR_PIXELL(4, 4)
PLANAR_PIXELL(5, 5)
PLANAR_PIXELL(6, 6)
PLANAR_PIXELL(7, 7)
}
static void TileFlipM4Low(int sx, unsigned int pack, int pal)
{
unsigned char *pd = Pico.est.HighCol + sx;
unsigned int t;
PLANAR_PIXELL(0, 7)
PLANAR_PIXELL(1, 6)
PLANAR_PIXELL(2, 5)
PLANAR_PIXELL(3, 4)
PLANAR_PIXELL(4, 3)
PLANAR_PIXELL(5, 2)
PLANAR_PIXELL(6, 1)
PLANAR_PIXELL(7, 0)
}
#define PLANAR_PIXEL(x,p) \ #define PLANAR_PIXEL(x,p) \
t = pack & (0x80808080 >> p); \ t = pack & (0x80808080 >> p); \
if (t) { \ if (t) { \
@ -111,7 +146,49 @@ static void draw_sprites(int scanline)
} }
// tilex_ty_prio merged to reduce register pressure // tilex_ty_prio merged to reduce register pressure
static void draw_strip(const unsigned short *nametab, int dx, int cells, int tilex_ty_prio) static void draw_strip_low(const unsigned short *nametab, int dx, int cells, int tilex_ty_prio)
{
int oldcode = -1, blank = -1; // The tile we know is blank
int addr = 0, pal = 0;
// Draw tiles across screen:
for (; cells > 0; dx += 8, tilex_ty_prio++, cells--)
{
unsigned int pack;
int code;
code = nametab[tilex_ty_prio & 0x1f];
if (code == blank)
continue;
/*
if ((code ^ tilex_ty_prio) & 0x1000) // priority differs?
continue;
*/
if (code != oldcode) {
oldcode = code;
// Get tile address/2:
addr = (code & 0x1ff) << 4;
addr += tilex_ty_prio >> 16;
if (code & 0x0400)
addr ^= 0xe; // Y-flip
pal = (code>>7) & 0x10;
}
pack = *(unsigned int *)(PicoMem.vram + addr); /* Get 4 bitplanes / 8 pixels */
/*
if (pack == 0) {
blank = code;
continue;
}
*/
if (code & 0x0200) TileFlipM4Low(dx, pack, pal);
else TileNormM4Low(dx, pack, pal);
}
}
// tilex_ty_prio merged to reduce register pressure
static void draw_strip_high(const unsigned short *nametab, int dx, int cells, int tilex_ty_prio)
{ {
int oldcode = -1, blank = -1; // The tile we know is blank int oldcode = -1, blank = -1; // The tile we know is blank
int addr = 0, pal = 0; int addr = 0, pal = 0;
@ -184,7 +261,7 @@ static void DrawDisplayM4(int scanline)
// low priority tiles // low priority tiles
if (!(pv->debug_p & PVD_KILL_B)) if (!(pv->debug_p & PVD_KILL_B))
draw_strip(nametab, dx, cells, tilex | 0x0000 | (ty << 16)); draw_strip_low(nametab, dx, cells, tilex | 0x0000 | (ty << 16));
// sprites // sprites
if (!(pv->debug_p & PVD_KILL_S_LO)) if (!(pv->debug_p & PVD_KILL_S_LO))
@ -192,7 +269,7 @@ static void DrawDisplayM4(int scanline)
// high priority tiles (use virtual layer switch just for fun) // high priority tiles (use virtual layer switch just for fun)
if (!(pv->debug_p & PVD_KILL_A)) if (!(pv->debug_p & PVD_KILL_A))
draw_strip(nametab, dx, cells, tilex | 0x1000 | (ty << 16)); draw_strip_high(nametab, dx, cells, tilex | 0x1000 | (ty << 16));
if (pv->reg[0] & 0x20) { if (pv->reg[0] & 0x20) {
// first column masked, caculate offset to start of line // first column masked, caculate offset to start of line

View file

@ -38,6 +38,7 @@ void PicoInit(void)
PicoInitMCD(); PicoInitMCD();
PicoSVPInit(); PicoSVPInit();
Pico32xInit(); Pico32xInit();
PsndInit();
PicoDrawInit(); PicoDrawInit();
PicoDraw2Init(); PicoDraw2Init();
@ -50,6 +51,7 @@ void PicoExit(void)
PicoExitMCD(); PicoExitMCD();
PicoCartUnload(); PicoCartUnload();
z80_exit(); z80_exit();
PsndExit();
free(Pico.sv.data); free(Pico.sv.data);
Pico.sv.data = NULL; Pico.sv.data = NULL;

View file

@ -55,7 +55,7 @@ extern void *p32x_bios_g, *p32x_bios_m, *p32x_bios_s;
#define POPT_EN_Z80 (1<< 2) #define POPT_EN_Z80 (1<< 2)
#define POPT_EN_STEREO (1<< 3) #define POPT_EN_STEREO (1<< 3)
#define POPT_ALT_RENDERER (1<< 4) // 00 00x0 #define POPT_ALT_RENDERER (1<< 4) // 00 00x0
// unused (1<< 5) #define POPT_EN_YM2413 (1<< 5)
// unused (1<< 6) // unused (1<< 6)
#define POPT_ACC_SPRITES (1<< 7) #define POPT_ACC_SPRITES (1<< 7)
#define POPT_DIS_32C_BORDER (1<< 8) // 00 0x00 #define POPT_DIS_32C_BORDER (1<< 8) // 00 0x00

View file

@ -434,6 +434,7 @@ struct PicoSound
unsigned int dac_pos; // last DAC position in Q20 unsigned int dac_pos; // last DAC position in Q20
unsigned int fm_pos; // last FM position in Q20 unsigned int fm_pos; // last FM position in Q20
unsigned int psg_pos; // last PSG position in Q16 unsigned int psg_pos; // last PSG position in Q16
unsigned int ym2413_pos; // last YM2413 position
}; };
// run tools/mkoffsets pico/pico_int_offs.h if you change these // run tools/mkoffsets pico/pico_int_offs.h if you change these
@ -897,10 +898,13 @@ PICO_INTERNAL_ASM void wram_2M_to_1M(unsigned char *m);
PICO_INTERNAL_ASM void wram_1M_to_2M(unsigned char *m); PICO_INTERNAL_ASM void wram_1M_to_2M(unsigned char *m);
// sound/sound.c // sound/sound.c
PICO_INTERNAL void PsndInit(void);
PICO_INTERNAL void PsndExit(void);
PICO_INTERNAL void PsndReset(void); PICO_INTERNAL void PsndReset(void);
PICO_INTERNAL void PsndStartFrame(void); PICO_INTERNAL void PsndStartFrame(void);
PICO_INTERNAL void PsndDoDAC(int cycle_to); PICO_INTERNAL void PsndDoDAC(int cycle_to);
PICO_INTERNAL void PsndDoPSG(int line_to); PICO_INTERNAL void PsndDoPSG(int line_to);
PICO_INTERNAL void PsndDoYM2413(int line_to);
PICO_INTERNAL void PsndDoFM(int line_to); PICO_INTERNAL void PsndDoFM(int line_to);
PICO_INTERNAL void PsndClear(void); PICO_INTERNAL void PsndClear(void);
PICO_INTERNAL void PsndGetSamples(int y); PICO_INTERNAL void PsndGetSamples(int y);

View file

@ -15,6 +15,13 @@
#include "pico_int.h" #include "pico_int.h"
#include "memory.h" #include "memory.h"
#include "sound/sn76496.h" #include "sound/sn76496.h"
#include "sound/emu2413/emu2413.h"
extern void YM2413_regWrite(unsigned reg);
extern void YM2413_dataWrite(unsigned data);
static unsigned short ymflag = 0xffff;
static unsigned char vdp_data_read(void) static unsigned char vdp_data_read(void)
{ {
@ -100,42 +107,61 @@ static unsigned char z80_sms_in(unsigned short a)
unsigned char d = 0; unsigned char d = 0;
elprintf(EL_IO, "z80 port %04x read", a); elprintf(EL_IO, "z80 port %04x read", a);
a &= 0xc1; if((a&0xff)>= 0xf0){
switch (a) switch((a&0xff))
{ {
case 0x00: case 0xf0:
case 0x01: // FM reg port
d = 0xff;
break; break;
case 0xf1:
case 0x40: /* V counter */ // FM data port
d = Pico.video.v_counter;
elprintf(EL_HVCNT, "V counter read: %02x", d);
break; break;
case 0xf2:
case 0x41: /* H counter */ // bit 0 = 1 active FM Pac
d = Pico.m.rotate++; if (PicoIn.opt & POPT_EN_YM2413){
elprintf(EL_HVCNT, "H counter read: %02x", d); d = ymflag;
break; //printf("read FM Check = %02x\n", d);
}
case 0x80:
d = vdp_data_read();
break;
case 0x81:
d = vdp_ctl_read();
break;
case 0xc0: /* I/O port A and B */
d = ~((PicoIn.pad[0] & 0x3f) | (PicoIn.pad[1] << 6));
break;
case 0xc1: /* I/O port B and miscellaneous */
d = (Pico.ms.io_ctl & 0x80) | ((Pico.ms.io_ctl << 1) & 0x40) | 0x30;
d |= ~(PicoIn.pad[1] >> 2) & 0x0f;
break; break;
}
} }
else{
a &= 0xc1;
switch (a)
{
case 0x00:
case 0x01:
d = 0xff;
break;
case 0x40: /* V counter */
d = Pico.video.v_counter;
elprintf(EL_HVCNT, "V counter read: %02x", d);
break;
case 0x41: /* H counter */
d = Pico.m.rotate++;
elprintf(EL_HVCNT, "H counter read: %02x", d);
break;
case 0x80:
d = vdp_data_read();
break;
case 0x81:
d = vdp_ctl_read();
break;
case 0xc0: /* I/O port A and B */
d = ~((PicoIn.pad[0] & 0x3f) | (PicoIn.pad[1] << 6));
break;
case 0xc1: /* I/O port B and miscellaneous */
d = (Pico.ms.io_ctl & 0x80) | ((Pico.ms.io_ctl << 1) & 0x40) | 0x30;
d |= ~(PicoIn.pad[1] >> 2) & 0x0f;
break;
}
}
elprintf(EL_IO, "ret = %02x", d); elprintf(EL_IO, "ret = %02x", d);
return d; return d;
} }
@ -143,27 +169,52 @@ static unsigned char z80_sms_in(unsigned short a)
static void z80_sms_out(unsigned short a, unsigned char d) static void z80_sms_out(unsigned short a, unsigned char d)
{ {
elprintf(EL_IO, "z80 port %04x write %02x", a, d); elprintf(EL_IO, "z80 port %04x write %02x", a, d);
a &= 0xc1;
switch (a)
{
case 0x01:
Pico.ms.io_ctl = d;
break;
case 0x40: if((a&0xff)>= 0xf0){
case 0x41: switch((a&0xff))
if ((d & 0x90) == 0x90) {
PsndDoPSG(Pico.m.scanline); case 0xf0:
SN76496Write(d); // FM reg port
break; YM2413_regWrite(d);
//printf("write FM register = %02x\n", d);
break;
case 0xf1:
// FM data port
YM2413_dataWrite(d);
//printf("write FM data = %02x\n", d);
break;
case 0xf2:
// bit 0 = 1 active FM Pac
if (PicoIn.opt & POPT_EN_YM2413){
ymflag = d;
//printf("write FM Check = %02x\n", d);
}
break;
}
}
else{
a &= 0xc1;
switch (a)
{
case 0x01:
Pico.ms.io_ctl = d;
break;
case 0x80: case 0x40:
vdp_data_write(d); case 0x41:
break; if ((d & 0x90) == 0x90)
PsndDoPSG(Pico.m.scanline);
SN76496Write(d);
break;
case 0x81: case 0x80:
vdp_ctl_write(d); vdp_data_write(d);
break; break;
case 0x81:
vdp_ctl_write(d);
break;
}
} }
} }
@ -212,6 +263,7 @@ void PicoResetMS(void)
{ {
z80_reset(); z80_reset();
PsndReset(); // pal must be known here PsndReset(); // pal must be known here
ymflag = 0xffff;
} }
void PicoPowerMS(void) void PicoPowerMS(void)

1
pico/sound/emu2413 Submodule

@ -0,0 +1 @@
Subproject commit 9f1dcf848d0e33e775e49352f7bc83a9c0e87a81

View file

@ -13,6 +13,7 @@
#include "../pico_int.h" #include "../pico_int.h"
#include "../cd/cue.h" #include "../cd/cue.h"
#include "mix.h" #include "mix.h"
#include "emu2413/emu2413.h"
void (*PsndMix_32_to_16l)(short *dest, int *src, int count) = mix_32_to_16l_stereo; void (*PsndMix_32_to_16l)(short *dest, int *src, int count) = mix_32_to_16l_stereo;
@ -25,6 +26,25 @@ short cdda_out_buffer[2*1152];
// sn76496 // sn76496
extern int *sn76496_regs; extern int *sn76496_regs;
// ym2413
#define YM2413_CLK 3579545
OPLL old_opll;
static OPLL *opll = NULL;
unsigned YM2413_reg;
PICO_INTERNAL void PsndInit(void)
{
opll = OPLL_new(YM2413_CLK, PicoIn.sndRate);
OPLL_setChipType(opll,0);
OPLL_reset(opll);
}
PICO_INTERNAL void PsndExit(void)
{
OPLL_delete(opll);
opll = NULL;
}
PICO_INTERNAL void PsndReset(void) PICO_INTERNAL void PsndReset(void)
{ {
@ -59,6 +79,12 @@ void PsndRerate(int preserve_state)
SN76496_init(Pico.m.pal ? OSC_PAL/15 : OSC_NTSC/15, PicoIn.sndRate); SN76496_init(Pico.m.pal ? OSC_PAL/15 : OSC_NTSC/15, PicoIn.sndRate);
if (preserve_state) memcpy(sn76496_regs, state, 28*4); // restore old state if (preserve_state) memcpy(sn76496_regs, state, 28*4); // restore old state
if(opll != NULL){
if (preserve_state) memcpy(&old_opll, opll, sizeof(OPLL)); // remember old state
OPLL_setRate(opll, PicoIn.sndRate);
OPLL_reset(opll);
}
if (state) if (state)
free(state); free(state);
@ -161,6 +187,48 @@ PICO_INTERNAL void PsndDoPSG(int line_to)
SN76496Update(PicoIn.sndOut + pos, len, stereo); SN76496Update(PicoIn.sndOut + pos, len, stereo);
} }
#if 0
PICO_INTERNAL void PsndDoYM2413(int line_to)
{
int pos, len;
int stereo = 0;
short *buf;
// Q16, number of samples since last call
len = ((line_to+1) * Pico.snd.smpl_mult) - Pico.snd.ym2413_pos;
if (len <= 0)
return;
// update position and calculate buffer offset and length
pos = (Pico.snd.ym2413_pos+0x8000) >> 16;
Pico.snd.ym2413_pos += len;
len = ((Pico.snd.ym2413_pos+0x8000) >> 16) - pos;
if (!PicoIn.sndOut || !(PicoIn.opt & POPT_EN_YM2413))
return;
if (PicoIn.opt & POPT_EN_STEREO) {
stereo = 1;
pos <<= 1;
}
buf = PicoIn.sndOut + pos;
while (len-- > 0) {
int16_t getdata = OPLL_calc(opll) * 3;
*buf++ += getdata;
buf += stereo; // only left for stereo, to be mixed to right later
}
}
#endif
void YM2413_regWrite(unsigned data){
OPLL_writeIO(opll,0,data);
}
void YM2413_dataWrite(unsigned data){
OPLL_writeIO(opll,1,data);
}
PICO_INTERNAL void PsndDoFM(int cyc_to) PICO_INTERNAL void PsndDoFM(int cyc_to)
{ {
int pos, len; int pos, len;
@ -249,7 +317,7 @@ PICO_INTERNAL void PsndClear(void)
if (!(PicoIn.opt & POPT_EN_FM)) if (!(PicoIn.opt & POPT_EN_FM))
memset32(PsndBuffer, 0, PicoIn.opt & POPT_EN_STEREO ? len*2 : len); memset32(PsndBuffer, 0, PicoIn.opt & POPT_EN_STEREO ? len*2 : len);
// drop pos remainder to avoid rounding errors (not entirely correct though) // drop pos remainder to avoid rounding errors (not entirely correct though)
Pico.snd.dac_pos = Pico.snd.fm_pos = Pico.snd.psg_pos = 0; Pico.snd.dac_pos = Pico.snd.fm_pos = Pico.snd.psg_pos = Pico.snd.ym2413_pos = 0;
} }
@ -344,6 +412,7 @@ static int PsndRenderMS(int offset, int length)
{ {
int stereo = (PicoIn.opt & 8) >> 3; int stereo = (PicoIn.opt & 8) >> 3;
int psglen = ((Pico.snd.psg_pos+0x8000) >> 16); int psglen = ((Pico.snd.psg_pos+0x8000) >> 16);
int ym2413len = ((Pico.snd.ym2413_pos+0x8000) >> 16);
pprof_start(sound); pprof_start(sound);
@ -355,11 +424,25 @@ static int PsndRenderMS(int offset, int length)
SN76496Update(psgbuf, length-psglen, stereo); SN76496Update(psgbuf, length-psglen, stereo);
} }
if (length-ym2413len > 0) {
short *ym2413buf = PicoIn.sndOut + (ym2413len << stereo);
Pico.snd.ym2413_pos += (length-ym2413len) << 16;
int len = (length-ym2413len);
if (PicoIn.opt & POPT_EN_YM2413){
while (len-- > 0) {
int16_t getdata = OPLL_calc(opll) * 3;
*ym2413buf += getdata;
ym2413buf += 1<<stereo;
}
}
}
// upmix to "stereo" if needed // upmix to "stereo" if needed
if (PicoIn.opt & POPT_EN_STEREO) { if (PicoIn.opt & POPT_EN_STEREO) {
int i, *p; int i;
for (i = length, p = (void *)PicoIn.sndOut; i > 0; i--, p++) short *p;
*p |= *p << 16; for (i = length, p = (short *)PicoIn.sndOut; i > 0; i--, p+=2)
*(p + 1) = *p;
} }
pprof_end(sound); pprof_end(sound);

View file

@ -11,10 +11,12 @@
#include "../cpu/sh2/sh2.h" #include "../cpu/sh2/sh2.h"
#include "sound/ym2612.h" #include "sound/ym2612.h"
#include "sound/emu2413/emu2413.h"
#include "state.h" #include "state.h"
// sn76496 // sn76496 & ym2413
extern int *sn76496_regs; extern int *sn76496_regs;
extern OPLL old_opll;
static arearw *areaRead; static arearw *areaRead;
static arearw *areaWrite; static arearw *areaWrite;
@ -123,6 +125,8 @@ typedef enum {
CHUNK_DRAM, CHUNK_DRAM,
CHUNK_32XPAL, CHUNK_32XPAL,
CHUNK_32X_EVT, CHUNK_32X_EVT,
CHUNK_YM2413, //40
//rename
CHUNK_32X_FIRST = CHUNK_MSH2, CHUNK_32X_FIRST = CHUNK_MSH2,
CHUNK_32X_LAST = CHUNK_32X_EVT, CHUNK_32X_LAST = CHUNK_32X_EVT,
// add new stuff here // add new stuff here
@ -133,6 +137,7 @@ typedef enum {
// //
CHUNK_DEFAULT_COUNT, CHUNK_DEFAULT_COUNT,
CHUNK_CARTHW_ = CHUNK_CARTHW, // 64 (defined in PicoInt) CHUNK_CARTHW_ = CHUNK_CARTHW, // 64 (defined in PicoInt)
} chunk_name_e; } chunk_name_e;
static const char * const chunk_names[CHUNK_DEFAULT_COUNT] = { static const char * const chunk_names[CHUNK_DEFAULT_COUNT] = {
@ -179,6 +184,7 @@ static const char * const chunk_names[CHUNK_DEFAULT_COUNT] = {
"DRAM", "DRAM",
"PAL", "PAL",
"events", "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)
@ -283,6 +289,8 @@ static int state_save(void *file)
memcpy(buff, pcd_event_times, sizeof(pcd_event_times)); memcpy(buff, pcd_event_times, sizeof(pcd_event_times));
CHECKED_WRITE(CHUNK_CD_EVT, 0x40, buff); CHECKED_WRITE(CHUNK_CD_EVT, 0x40, buff);
CHECKED_WRITE(CHUNK_YM2413, sizeof(OPLL), &old_opll);
len = gfx_context_save(buf2); len = gfx_context_save(buf2);
CHECKED_WRITE(CHUNK_CD_GFX, len, buf2); CHECKED_WRITE(CHUNK_CD_GFX, len, buf2);
len = cdc_context_save(buf2); len = cdc_context_save(buf2);
@ -442,6 +450,7 @@ static int state_load(void *file)
case CHUNK_IOPORTS: CHECKED_READ_BUFF(PicoMem.ioports); break; case CHUNK_IOPORTS: CHECKED_READ_BUFF(PicoMem.ioports); break;
case CHUNK_PSG: CHECKED_READ2(28*4, sn76496_regs); break; case CHUNK_PSG: CHECKED_READ2(28*4, sn76496_regs); break;
case CHUNK_YM2413: CHECKED_READ2(sizeof(OPLL), &old_opll); break;
case CHUNK_FM: case CHUNK_FM:
ym2612_regs = YM2612GetRegs(); ym2612_regs = YM2612GetRegs();
CHECKED_READ2(0x200+4, ym2612_regs); CHECKED_READ2(0x200+4, ym2612_regs);

View file

@ -124,6 +124,7 @@ endif
# sound # sound
SRCS_COMMON += $(R)pico/sound/sound.c SRCS_COMMON += $(R)pico/sound/sound.c
SRCS_COMMON += $(R)pico/sound/sn76496.c $(R)pico/sound/ym2612.c SRCS_COMMON += $(R)pico/sound/sn76496.c $(R)pico/sound/ym2612.c
SRCS_COMMON += $(R)pico/sound/emu2413/emu2413.c
ifneq "$(ARCH)$(asm_mix)" "arm1" ifneq "$(ARCH)$(asm_mix)" "arm1"
SRCS_COMMON += $(R)pico/sound/mix.c SRCS_COMMON += $(R)pico/sound/mix.c
endif endif

View file

@ -501,6 +501,7 @@ static menu_entry e_menu_adv_options[] =
mee_onoff ("Emulate YM2612 (FM)", MA_OPT2_ENABLE_YM2612, PicoIn.opt, POPT_EN_FM), mee_onoff ("Emulate YM2612 (FM)", MA_OPT2_ENABLE_YM2612, PicoIn.opt, POPT_EN_FM),
mee_onoff ("Disable YM2612 SSG-EG", MA_OPT2_DISABLE_YM_SSG,PicoIn.opt, POPT_DIS_FM_SSGEG), mee_onoff ("Disable YM2612 SSG-EG", MA_OPT2_DISABLE_YM_SSG,PicoIn.opt, POPT_DIS_FM_SSGEG),
mee_onoff ("Emulate SN76496 (PSG)", MA_OPT2_ENABLE_SN76496,PicoIn.opt, POPT_EN_PSG), mee_onoff ("Emulate SN76496 (PSG)", MA_OPT2_ENABLE_SN76496,PicoIn.opt, POPT_EN_PSG),
mee_onoff ("Emulate YM2413 (FM)", MA_OPT2_ENABLE_YM2413 ,PicoIn.opt, POPT_EN_YM2413),
mee_onoff ("gzip savestates", MA_OPT2_GZIP_STATES, currentConfig.EmuOpt, EOPT_GZIP_SAVES), mee_onoff ("gzip savestates", MA_OPT2_GZIP_STATES, currentConfig.EmuOpt, EOPT_GZIP_SAVES),
mee_onoff ("Don't save last used ROM", MA_OPT2_NO_LAST_ROM, currentConfig.EmuOpt, EOPT_NO_AUTOSVCFG), mee_onoff ("Don't save last used ROM", MA_OPT2_NO_LAST_ROM, currentConfig.EmuOpt, EOPT_NO_AUTOSVCFG),
mee_onoff ("Disable idle loop patching",MA_OPT2_NO_IDLE_LOOPS,PicoIn.opt, POPT_DIS_IDLE_DET), mee_onoff ("Disable idle loop patching",MA_OPT2_NO_IDLE_LOOPS,PicoIn.opt, POPT_DIS_IDLE_DET),

View file

@ -50,6 +50,7 @@ typedef enum
MA_OPT2_ENABLE_YM2612, MA_OPT2_ENABLE_YM2612,
MA_OPT2_DISABLE_YM_SSG, MA_OPT2_DISABLE_YM_SSG,
MA_OPT2_ENABLE_SN76496, MA_OPT2_ENABLE_SN76496,
MA_OPT2_ENABLE_YM2413,
MA_OPT2_GZIP_STATES, MA_OPT2_GZIP_STATES,
MA_OPT2_NO_LAST_ROM, MA_OPT2_NO_LAST_ROM,
MA_OPT2_RAMTIMINGS, /* gp2x */ MA_OPT2_RAMTIMINGS, /* gp2x */

View file

@ -64,6 +64,7 @@ OBJS += pico/sound/sound.o
endif endif
OBJS += pico/sound/mix_asm.o OBJS += pico/sound/mix_asm.o
OBJS += pico/sound/sn76496.o pico/sound/ym2612.o OBJS += pico/sound/sn76496.o pico/sound/ym2612.o
OBJS += pico/sound/emu2413/emu2413.o
# zlib # zlib
OBJS += zlib/gzio.o zlib/inffast.o zlib/inflate.o zlib/inftrees.o zlib/trees.o \ OBJS += zlib/gzio.o zlib/inffast.o zlib/inflate.o zlib/inftrees.o zlib/trees.o \
zlib/deflate.o zlib/crc32.o zlib/adler32.o zlib/zutil.o zlib/compress.o zlib/deflate.o zlib/crc32.o zlib/adler32.o zlib/zutil.o zlib/compress.o

View file

@ -931,6 +931,7 @@ menu_entry opt2_entries[] =
{ "Emulate Z80", MB_ONOFF, MA_OPT2_ENABLE_Z80, &PicoIn.opt, 0x00004, 0, 0, 1, 1 }, { "Emulate Z80", MB_ONOFF, MA_OPT2_ENABLE_Z80, &PicoIn.opt, 0x00004, 0, 0, 1, 1 },
{ "Emulate YM2612 (FM)", MB_ONOFF, MA_OPT2_ENABLE_YM2612, &PicoIn.opt, 0x00001, 0, 0, 1, 1 }, { "Emulate YM2612 (FM)", MB_ONOFF, MA_OPT2_ENABLE_YM2612, &PicoIn.opt, 0x00001, 0, 0, 1, 1 },
{ "Emulate SN76496 (PSG)", MB_ONOFF, MA_OPT2_ENABLE_SN76496,&PicoIn.opt, 0x00002, 0, 0, 1, 1 }, { "Emulate SN76496 (PSG)", MB_ONOFF, MA_OPT2_ENABLE_SN76496,&PicoIn.opt, 0x00002, 0, 0, 1, 1 },
{ "Emulate YM2413 (FM)", MB_ONOFF, MA_OPT2_ENABLE_YM2413, &PicoIn.opt, 0x00020, 0, 0, 1, 1 },
{ "Double buffering", MB_ONOFF, MA_OPT2_DBLBUFF, &currentConfig.EmuOpt, 0x8000, 0, 0, 1, 1 }, { "Double buffering", MB_ONOFF, MA_OPT2_DBLBUFF, &currentConfig.EmuOpt, 0x8000, 0, 0, 1, 1 },
{ "Wait for V-sync (slow)", MB_ONOFF, MA_OPT2_VSYNC, &currentConfig.EmuOpt, 0x2000, 0, 0, 1, 1 }, { "Wait for V-sync (slow)", MB_ONOFF, MA_OPT2_VSYNC, &currentConfig.EmuOpt, 0x2000, 0, 0, 1, 1 },
{ "gzip savestates", MB_ONOFF, MA_OPT2_GZIP_STATES, &currentConfig.EmuOpt, 0x0008, 0, 0, 1, 1 }, { "gzip savestates", MB_ONOFF, MA_OPT2_GZIP_STATES, &currentConfig.EmuOpt, 0x0008, 0, 0, 1, 1 },

View file

@ -1366,7 +1366,7 @@ void retro_init(void)
sceBlock = getVMBlock(); sceBlock = getVMBlock();
#endif #endif
PicoIn.opt = POPT_EN_STEREO|POPT_EN_FM|POPT_EN_PSG|POPT_EN_Z80 PicoIn.opt = POPT_EN_STEREO|POPT_EN_FM|POPT_EN_PSG|POPT_EN_Z80|POPT_EN_YM2413
| POPT_EN_MCD_PCM|POPT_EN_MCD_CDDA|POPT_EN_MCD_GFX | POPT_EN_MCD_PCM|POPT_EN_MCD_CDDA|POPT_EN_MCD_GFX
| POPT_EN_32X|POPT_EN_PWM | POPT_EN_32X|POPT_EN_PWM
| POPT_ACC_SPRITES|POPT_DIS_32C_BORDER; | POPT_ACC_SPRITES|POPT_DIS_32C_BORDER;

View file

@ -1,9 +1,9 @@
[Desktop Entry] [Desktop Entry]
Type=Application
Name=Picodrive Name=Picodrive
Comment=A megadrive/genesis emulator Comment=A megadrive/genesis emulator
Exec=PicoDrive Exec=PicoDrive %f
Terminal=false
Type=Application
StartupNotify=true
Icon=megadrive Icon=megadrive
Terminal=false
Categories=emulators; Categories=emulators;
MimeType=.md;.smd;.bin;.sms;.cue;.32x;.zip;.7z

View file

@ -37,6 +37,7 @@ const struct menu_keymap in_sdl_key_map[] =
{ SDLK_BACKSPACE, PBTN_R }, { SDLK_BACKSPACE, PBTN_R },
}; };
#if !defined(__RG350__)
const char * const in_sdl_key_names[SDLK_LAST] = { const char * const in_sdl_key_names[SDLK_LAST] = {
[SDLK_UP] = "UP", [SDLK_UP] = "UP",
[SDLK_DOWN] = "DOWN", [SDLK_DOWN] = "DOWN",
@ -52,4 +53,33 @@ const char * const in_sdl_key_names[SDLK_LAST] = {
[SDLK_ESCAPE] = "SELECT", [SDLK_ESCAPE] = "SELECT",
[SDLK_POWER] = "POWER", [SDLK_POWER] = "POWER",
[SDLK_PAUSE] = "LOCK", [SDLK_PAUSE] = "LOCK",
[SDLK_PAGEUP] = "L2",
[SDLK_PAGEDOWN] = "R2",
[SDLK_KP_DIVIDE] = "L3",
[SDLK_KP_PERIOD] = "R3",
}; };
#else
/* RG 350 */
const char * const in_sdl_key_names[SDLK_LAST] = {
[SDLK_UP] = "UP",
[SDLK_DOWN] = "DOWN",
[SDLK_LEFT] = "LEFT",
[SDLK_RIGHT] = "RIGHT",
[SDLK_LCTRL] = "A",
[SDLK_LALT] = "B",
[SDLK_SPACE] = "X",
[SDLK_LSHIFT] = "Y",
[SDLK_TAB] = "L",
[SDLK_BACKSPACE] = "R",
[SDLK_RETURN] = "START",
[SDLK_ESCAPE] = "SELECT",
[SDLK_HOME] = "POWER",
[SDLK_PAUSE] = "LOCK",
[SDLK_PAGEUP] = "L2",
[SDLK_PAGEDOWN] = "R2",
[SDLK_KP_DIVIDE] = "L3",
[SDLK_KP_PERIOD] = "R3",
};
#endif

View file

@ -1119,6 +1119,7 @@ menu_entry opt2_entries[] =
{ "Emulate Z80", MB_ONOFF, MA_OPT2_ENABLE_Z80, &PicoIn.opt, 0x00004, 0, 0, 1, 1 }, { "Emulate Z80", MB_ONOFF, MA_OPT2_ENABLE_Z80, &PicoIn.opt, 0x00004, 0, 0, 1, 1 },
{ "Emulate YM2612 (FM)", MB_ONOFF, MA_OPT2_ENABLE_YM2612, &PicoIn.opt, 0x00001, 0, 0, 1, 1 }, { "Emulate YM2612 (FM)", MB_ONOFF, MA_OPT2_ENABLE_YM2612, &PicoIn.opt, 0x00001, 0, 0, 1, 1 },
{ "Emulate SN76496 (PSG)", MB_ONOFF, MA_OPT2_ENABLE_SN76496, &PicoIn.opt, 0x00002, 0, 0, 1, 1 }, { "Emulate SN76496 (PSG)", MB_ONOFF, MA_OPT2_ENABLE_SN76496, &PicoIn.opt, 0x00002, 0, 0, 1, 1 },
{ "Emulate YM2413 (FM)", MB_ONOFF, MA_OPT2_ENABLE_YM2413, &PicoIn.opt, 0x00020, 0, 0, 1, 1 },
{ "gzip savestates", MB_ONOFF, MA_OPT2_GZIP_STATES, &currentConfig.EmuOpt, 0x00008, 0, 0, 1, 1 }, { "gzip savestates", MB_ONOFF, MA_OPT2_GZIP_STATES, &currentConfig.EmuOpt, 0x00008, 0, 0, 1, 1 },
{ "Don't save last used ROM", MB_ONOFF, MA_OPT2_NO_LAST_ROM, &currentConfig.EmuOpt, 0x00020, 0, 0, 1, 1 }, { "Don't save last used ROM", MB_ONOFF, MA_OPT2_NO_LAST_ROM, &currentConfig.EmuOpt, 0x00020, 0, 0, 1, 1 },
{ "Status line in main menu", MB_ONOFF, MA_OPT2_STATUS_LINE, &currentConfig.EmuOpt, 0x20000, 0, 0, 1, 1 }, { "Status line in main menu", MB_ONOFF, MA_OPT2_STATUS_LINE, &currentConfig.EmuOpt, 0x20000, 0, 0, 1, 1 },