From 73ba5e9a618e814cce83a0ae0433b1da4e48df10 Mon Sep 17 00:00:00 2001 From: libretro Date: Tue, 13 Nov 2012 18:35:39 +0000 Subject: [PATCH] Initial libretro commit - had to rename user-defined headers which are named after system headers, ie. memory.h, types.h, endian.h --- mednafen/libmednahawk/libmednahawk.vcxproj | 6 +- .../libmednahawk/libmednahawk.vcxproj.filters | 6 +- mednafen/libretro/Makefile | 325 ++++++ mednafen/libretro/README.md | 24 + mednafen/libretro/config.h | 24 +- mednafen/libretro/inttypes.h | 13 +- mednafen/libretro/libretro.cpp | 581 +++++++++++ mednafen/libretro/libretro.h | 577 +++++++++++ mednafen/libretro/link.T | 5 + mednafen/libretro/minifen/minifen.cpp | 963 ++++++++++++++++++ mednafen/libretro/minifen/minifen_state.cpp | 542 ++++++++++ mednafen/libretro/scrc32.cpp | 74 ++ mednafen/libretro/scrc32.h | 6 + mednafen/libretro/stubs.cpp | 165 +++ mednafen/libretro/thread.cpp | 295 ++++++ mednafen/libretro/thread.h | 46 + mednafen/libretro/thread/gx_pthread.h | 96 ++ mednafen/src/FileWrapper.cpp | 4 + mednafen/src/PSFLoader.cpp | 2 +- mednafen/src/cdrom/CDAccess_Image.cpp | 2 +- mednafen/src/cdrom/audioreader.cpp | 4 +- mednafen/src/cdrom/dvdisaster.h | 2 +- mednafen/src/driver.h | 2 +- mednafen/src/drivers/args.cpp | 2 +- mednafen/src/endian.cpp | 2 +- mednafen/src/md5.cpp | 2 +- mednafen/src/{endian.h => mednafen-endian.h} | 2 +- mednafen/src/{memory.h => mednafen-memory.h} | 0 mednafen/src/{types.h => mednafen-types.h} | 0 mednafen/src/mednafen.h | 10 +- mednafen/src/memory.cpp | 2 +- mednafen/src/pcfx/jrevdct.cpp | 2 +- mednafen/src/psx/cpu.cpp | 3 + mednafen/src/qtrecord.cpp | 2 +- mednafen/src/string/ConvertUTF.cpp | 2 +- mednafen/src/video/png.cpp | 2 +- mednafen/src/wswan/debug.cpp | 2 +- mednafen/src/wswan/dis/disasm.h | 2 +- mednafen/src/wswan/v30mz.cpp | 2 +- 39 files changed, 3757 insertions(+), 44 deletions(-) create mode 100644 mednafen/libretro/Makefile create mode 100644 mednafen/libretro/README.md create mode 100644 mednafen/libretro/libretro.cpp create mode 100755 mednafen/libretro/libretro.h create mode 100644 mednafen/libretro/link.T create mode 100644 mednafen/libretro/minifen/minifen.cpp create mode 100644 mednafen/libretro/minifen/minifen_state.cpp create mode 100644 mednafen/libretro/scrc32.cpp create mode 100644 mednafen/libretro/scrc32.h create mode 100644 mednafen/libretro/stubs.cpp create mode 100644 mednafen/libretro/thread.cpp create mode 100644 mednafen/libretro/thread.h create mode 100644 mednafen/libretro/thread/gx_pthread.h rename mednafen/src/{endian.h => mednafen-endian.h} (94%) rename mednafen/src/{memory.h => mednafen-memory.h} (100%) rename mednafen/src/{types.h => mednafen-types.h} (100%) diff --git a/mednafen/libmednahawk/libmednahawk.vcxproj b/mednafen/libmednahawk/libmednahawk.vcxproj index b623df7575..4080acf6a3 100644 --- a/mednafen/libmednahawk/libmednahawk.vcxproj +++ b/mednafen/libmednahawk/libmednahawk.vcxproj @@ -318,7 +318,7 @@ - + @@ -327,7 +327,7 @@ - + @@ -448,4 +448,4 @@ - \ No newline at end of file + diff --git a/mednafen/libmednahawk/libmednahawk.vcxproj.filters b/mednafen/libmednahawk/libmednahawk.vcxproj.filters index e191adbaa5..a739c05977 100644 --- a/mednafen/libmednahawk/libmednahawk.vcxproj.filters +++ b/mednafen/libmednahawk/libmednahawk.vcxproj.filters @@ -429,7 +429,7 @@ src - + src @@ -456,7 +456,7 @@ src - + src @@ -791,4 +791,4 @@ src\psx - \ No newline at end of file + diff --git a/mednafen/libretro/Makefile b/mednafen/libretro/Makefile new file mode 100644 index 0000000000..f1353585ba --- /dev/null +++ b/mednafen/libretro/Makefile @@ -0,0 +1,325 @@ +DEBUG = 0 +FRONTEND_SUPPORTS_RGB565 = 1 + +PREV_ROOT_DIR := ../ +MEDNAFEN_DIR := ../src + +ifeq ($(platform),) +platform = unix +ifeq ($(shell uname -a),) + platform = win +else ifneq ($(findstring Darwin,$(shell uname -a)),) + platform = osx +else ifneq ($(findstring MINGW,$(shell uname -a)),) + platform = win +endif +endif + +# If you have a system with 1GB RAM or more - cache the whole +# CD in order to prevent file access delays/hiccups +CACHE_CD = 0 + +#if no core specified, just pick psx for now +ifeq ($(core),) + core = psx +endif + +NEED_MINIFEN = 1 + +ifeq ($(core), psx) + core = psx + PTHREAD_FLAGS = -pthread + NEED_CD = 1 + NEED_THREADING = 1 + NEED_BPP = 32 + NEED_BLIP = 1 + NEED_DEINTERLACER = 1 + CORE_DEFINE := -DWANT_PSX_EMU + CORE_DIR := $(MEDNAFEN_DIR)/psx + CORE_SOURCES := $(CORE_DIR)/psx.cpp \ + $(CORE_DIR)/irq.cpp \ + $(CORE_DIR)/timer.cpp \ + $(CORE_DIR)/dma.cpp \ + $(CORE_DIR)/frontio.cpp \ + $(CORE_DIR)/sio.cpp \ + $(CORE_DIR)/cpu.cpp \ + $(CORE_DIR)/gte.cpp \ + $(CORE_DIR)/dis.cpp \ + $(CORE_DIR)/cdc.cpp \ + $(CORE_DIR)/spu.cpp \ + $(CORE_DIR)/gpu.cpp \ + $(CORE_DIR)/mdec.cpp \ + $(CORE_DIR)/input/gamepad.cpp \ + $(CORE_DIR)/input/dualanalog.cpp \ + $(CORE_DIR)/input/dualshock.cpp \ + $(CORE_DIR)/input/justifier.cpp \ + $(CORE_DIR)/input/guncon.cpp \ + $(CORE_DIR)/input/negcon.cpp \ + $(CORE_DIR)/input/memcard.cpp \ + $(CORE_DIR)/input/multitap.cpp \ + $(CORE_DIR)/input/mouse.cpp +TARGET_NAME := mednafen_psx_libretro +else ifeq ($(core), pce-fast) + core = pce_fast + PTHREAD_FLAGS = -pthread + NEED_BPP = 16 + NEED_BLIP = 1 + NEED_CD = 1 + NEED_CRC32 = 1 + NEED_THREADING = 1 + CORE_DEFINE := -DWANT_PCE_FAST_EMU + CORE_DIR := $(MEDNAFEN_DIR)/pce_fast-0924 + +CORE_SOURCES := $(CORE_DIR)/huc.cpp \ + $(CORE_DIR)/pce_huc6280.cpp \ + $(CORE_DIR)/input.cpp \ + $(CORE_DIR)/pce.cpp \ + $(CORE_DIR)/tsushin.cpp \ + $(CORE_DIR)/input/gamepad.cpp \ + $(CORE_DIR)/input/mouse.cpp \ + $(CORE_DIR)/input/tsushinkb.cpp \ + $(CORE_DIR)/vdc.cpp +TARGET_NAME := mednafen_pce_fast_libretro + +HW_CPU_SOURCES += $(MEDNAFEN_DIR)/hw_cpu/huc6280/huc6280.cpp +HW_MISC_SOURCES += $(MEDNAFEN_DIR)/hw_misc/arcade_card/arcade_card.cpp +HW_SOUND_SOURCES += $(MEDNAFEN_DIR)/hw_sound/pce_psg/pce_psg.cpp +HW_VIDEO_SOURCES += $(MEDNAFEN_DIR)/hw_video/huc6270/vdc.cpp +CDROM_SOURCES += $(MEDNAFEN_DIR)/cdrom/pcecd.cpp +else ifeq ($(core), wswan) + core = wswan + NEED_BPP = 16 + NEED_BLIP = 1 + CORE_DEFINE := -DWANT_WSWAN_EMU + CORE_DIR := $(MEDNAFEN_DIR)/wswan + +CORE_SOURCES := $(CORE_DIR)/gfx.cpp \ + $(CORE_DIR)/main.cpp \ + $(CORE_DIR)/memory.cpp \ + $(CORE_DIR)/v30mz.cpp \ + $(CORE_DIR)/sound.cpp \ + $(CORE_DIR)/tcache.cpp \ + $(CORE_DIR)/interrupt.cpp \ + $(CORE_DIR)/eeprom.cpp \ + $(CORE_DIR)/rtc.cpp +TARGET_NAME := mednafen_wswan_libretro +endif + +ifeq ($(NEED_RESAMPLER), 1) +FLAGS += -DNEED_RESAMPLER +RESAMPLER_SOURCES += $(MEDNAFEN_DIR)/sound/Fir_Resampler.cpp +ifeq ($(NEED_BLIP), 1) +RESAMPLER_SOURCES += $(MEDNAFEN_DIR)/sound/Blip_Buffer_orig.cpp +endif +else +ifeq ($(NEED_BLIP), 1) +RESAMPLER_SOURCES += $(MEDNAFEN_DIR)/sound/Blip_Buffer.cpp +endif +endif + +CORE_INCDIR := -I$(CORE_DIR) + +ifeq ($(platform), unix) + TARGET := $(TARGET_NAME).so + fpic := -fPIC + SHARED := -shared -Wl,--no-undefined -Wl,--version-script=link.T + ENDIANNESS_DEFINES := -DLSB_FIRST + ifneq ($(shell uname -p | grep -E '((i.|x)86|amd64)'),) + IS_X86 = 1 + endif + LDFLAGS += $(PTHREAD_FLAGS) + FLAGS += $(PTHREAD_FLAGS) +else ifeq ($(platform), osx) + TARGET := $(TARGET_NAME).dylib + fpic := -fPIC + SHARED := -dynamiclib + ENDIANNESS_DEFINES := -DLSB_FIRST + LDFLAGS += $(PTHREAD_FLAGS) + FLAGS += $(PTHREAD_FLAGS) +else ifeq ($(platform), ps3) + TARGET := $(TARGET_NAME)_ps3.a + CC = $(CELL_SDK)/host-win32/ppu/bin/ppu-lv2-gcc.exe + CXX = $(CELL_SDK)/host-win32/ppu/bin/ppu-lv2-g++.exe + AR = $(CELL_SDK)/host-win32/ppu/bin/ppu-lv2-ar.exe + ENDIANNESS_DEFINES := -DMSB_FIRST -DBYTE_ORDER=BIG_ENDIAN +else ifeq ($(platform), sncps3) + TARGET := $(TARGET_NAME)_ps3.a + CC = $(CELL_SDK)/host-win32/sn/bin/ps3ppusnc.exe + CXX = $(CELL_SDK)/host-win32/sn/bin/ps3ppusnc.exe + AR = $(CELL_SDK)/host-win32/sn/bin/ps3snarl.exe + ENDIANNESS_DEFINES := -DMSB_FIRST -DBYTE_ORDER=BIG_ENDIAN + + CXXFLAGS += -Xc+=exceptions +else ifeq ($(platform), psl1ght) + TARGET := $(TARGET_NAME)_psl1ght.a + CC = $(PS3DEV)/ppu/bin/ppu-gcc$(EXE_EXT) + CC = $(PS3DEV)/ppu/bin/ppu-g++$(EXE_EXT) + AR = $(PS3DEV)/ppu/bin/ppu-ar$(EXE_EXT) + ENDIANNESS_DEFINES := -DMSB_FIRST -DBYTE_ORDER=BIG_ENDIAN +else ifeq ($(platform), xenon) + TARGET := $(TARGET_NAME)_xenon360.a + CC = xenon-gcc$(EXE_EXT) + CXX = xenon-g++$(EXE_EXT) + AR = xenon-ar$(EXE_EXT) + ENDIANNESS_DEFINES += -D__LIBXENON__ -m32 -D__ppc__ -DMSB_FIRST -DBYTE_ORDER=BIG_ENDIAN + LIBS := $(PTHREAD_FLAGS) +else ifeq ($(platform), ngc) + TARGET := $(TARGET_NAME)_ngc.a + CC = $(DEVKITPPC)/bin/powerpc-eabi-gcc$(EXE_EXT) + CXX = $(DEVKITPPC)/bin/powerpc-eabi-g++$(EXE_EXT) + AR = $(DEVKITPPC)/bin/powerpc-eabi-ar$(EXE_EXT) + ENDIANNESS_DEFINES += -DGEKKO -DHW_DOL -mrvl -mcpu=750 -meabi -mhard-float -DMSB_FIRST -DBYTE_ORDER=BIG_ENDIAN + + EXTRA_INCLUDES := -I$(DEVKITPRO)/libogc/include + +else ifeq ($(platform), wii) + TARGET := $(TARGET_NAME)_wii.a + CC = $(DEVKITPPC)/bin/powerpc-eabi-gcc$(EXE_EXT) + CXX = $(DEVKITPPC)/bin/powerpc-eabi-g++$(EXE_EXT) + AR = $(DEVKITPPC)/bin/powerpc-eabi-ar$(EXE_EXT) + ENDIANNESS_DEFINES += -DGEKKO -DHW_RVL -mrvl -mcpu=750 -meabi -mhard-float -DMSB_FIRST -DBYTE_ORDER=BIG_ENDIAN + + EXTRA_INCLUDES := -I$(DEVKITPRO)/libogc/include +else + TARGET := retro.dll + CC = i686-pc-mingw32-gcc + CXX = i686-pc-mingw32-g++ + SHARED := -shared -Wl,--no-undefined -Wl,--version-script=link.T + LDFLAGS += -static-libgcc -static-libstdc++ -lwinmm + ENDIANNESS_DEFINES := -DLSB_FIRST +endif + +ifeq ($(NEED_THREADING), 1) +THREAD_STUBS += thread.cpp +endif + +ifeq ($(NEED_CRC32), 1) +FLAGS += -DHAVE_CRC32 +endif + +ifeq ($(NEED_DEINTERLACER), 1) +FLAGS += -DNEED_DEINTERLACER +endif + +ifeq ($(NEED_CD), 1) +CDROM_SOURCES += $(MEDNAFEN_DIR)/cdrom/CDAccess.cpp \ + $(MEDNAFEN_DIR)/cdrom/CDAccess_Image.cpp \ + $(MEDNAFEN_DIR)/cdrom/CDUtility.cpp \ + $(MEDNAFEN_DIR)/cdrom/lec.cpp \ + $(MEDNAFEN_DIR)/cdrom/SimpleFIFO.cpp \ + $(MEDNAFEN_DIR)/cdrom/audioreader.cpp \ + $(MEDNAFEN_DIR)/cdrom/galois.cpp \ + $(MEDNAFEN_DIR)/cdrom/pcecd.cpp \ + $(MEDNAFEN_DIR)/cdrom/scsicd.cpp \ + $(MEDNAFEN_DIR)/cdrom/recover-raw.cpp \ + $(MEDNAFEN_DIR)/cdrom/l-ec.cpp \ + $(MEDNAFEN_DIR)/cdrom/cdromif.cpp \ + $(MEDNAFEN_DIR)/cdrom/cd_crc32.cpp + +MPC_SRC := $(wildcard $(MEDNAFEN_DIR)/mpcdec/*.c) +TREMOR_SRC := $(wildcard $(MEDNAFEN_DIR)/tremor/*.c) +FLAGS += -DNEED_CD +endif + +ifeq ($(NEED_MINIFEN), 1) +MINIFEN_SOURCES += minifen/minifen.cpp minifen/minifen_state.cpp +FLAGS += -DNEED_MINIFEN +else +MINIFEN_SOURCES += $(MEDNAFEN_DIR)/mednafen.cpp $(MEDNAFEN_DIR)/state.cpp +endif + + +MEDNAFEN_SOURCES := $(MEDNAFEN_DIR)/error.cpp \ + $(MEDNAFEN_DIR)/math_ops.cpp \ + $(MEDNAFEN_DIR)/settings.cpp \ + $(MEDNAFEN_DIR)/general.cpp \ + $(MEDNAFEN_DIR)/FileWrapper.cpp \ + $(MEDNAFEN_DIR)/FileStream.cpp \ + $(MEDNAFEN_DIR)/MemoryStream.cpp \ + $(MEDNAFEN_DIR)/Stream.cpp \ + $(MEDNAFEN_DIR)/endian.cpp \ + $(CDROM_SOURCES) \ + $(MEDNAFEN_DIR)/mempatcher.cpp \ + $(MEDNAFEN_DIR)/video/video.cpp \ + $(MEDNAFEN_DIR)/video/Deinterlacer.cpp \ + $(MEDNAFEN_DIR)/video/surface.cpp \ + $(RESAMPLER_SOURCES) \ + $(MEDNAFEN_DIR)/sound/Stereo_Buffer.cpp \ + $(MEDNAFEN_DIR)/file.cpp \ + $(MEDNAFEN_DIR)/okiadpcm.cpp \ + $(MEDNAFEN_DIR)/md5.cpp + + +LIBRETRO_SOURCES := libretro.cpp stubs.cpp $(THREAD_STUBS) + +SOURCES_C := $(MEDNAFEN_DIR)/trio/trio.c \ + $(MPC_SRC) \ + $(TREMOR_SRC) \ + $(LIBRETRO_SOURCES_C) \ + $(MEDNAFEN_DIR)/trio/trionan.c \ + $(MEDNAFEN_DIR)/trio/triostr.c + +SOURCES := $(MINIFEN_SOURCES) $(LIBRETRO_SOURCES) $(CORE_SOURCES) $(MEDNAFEN_SOURCES) $(HW_CPU_SOURCES) $(HW_MISC_SOURCES) $(HW_SOUND_SOURCES) $(HW_VIDEO_SOURCES) + + +OBJECTS := $(SOURCES:.cpp=.o) $(SOURCES_C:.c=.o) + +all: $(TARGET) + +ifeq ($(DEBUG),0) + FLAGS += -O3 -ffast-math -funroll-loops +else + FLAGS += -O0 -g +endif + +LDFLAGS += $(fpic) $(SHARED) +FLAGS += -Wall $(fpic) -fno-strict-overflow +FLAGS += -I. -I$(PREV_ROOT_DIR)/include -I$(PREV_ROOT_DIR)/intl -I$(CORE_INCDIR) -I$(MEDNAFEN_DIR) + +WARNINGS := -Wall \ + -Wno-narrowing \ + -Wno-unused-but-set-variable \ + -Wno-sign-compare \ + -Wno-unused-variable \ + -Wno-unused-function \ + -Wno-uninitialized \ + -Wno-unused-result \ + -Wno-strict-aliasing \ + -Wno-overflow + +FLAGS += $(ENDIANNESS_DEFINES) -DSIZEOF_DOUBLE=8 $(WARNINGS) \ + -DMEDNAFEN_VERSION=\"0.9.26\" -DPACKAGE=\"mednafen\" -DMEDNAFEN_VERSION_NUMERIC=926 -DPSS_STYLE=1 -DMPC_FIXED_POINT -DARCH_X86 $(CORE_DEFINE) -DSTDC_HEADERS -D__STDC_LIMIT_MACROS -D__LIBRETRO__ -DNDEBUG -DHAVE_MKDIR $(EXTRA_INCLUDES) + +ifeq ($(CACHE_CD), 1) +FLAGS += -D__LIBRETRO_CACHE_CD__ +endif + +ifeq ($(NEED_BPP), 16) +FLAGS += -DWANT_16BPP +endif + +ifeq ($(FRONTEND_SUPPORTS_RGB565), 1) +FLAGS += -DFRONTEND_SUPPORTS_RGB565 +endif + +ifeq ($(NEED_BPP), 32) +FLAGS += -DWANT_32BPP +endif + + +CXXFLAGS += $(FLAGS) +CFLAGS += $(FLAGS) -std=gnu99 + +$(TARGET): $(OBJECTS) + $(CXX) -o $@ $^ $(LDFLAGS) + +%.o: %.cpp + $(CXX) -c -o $@ $< $(CXXFLAGS) + +%.o: %.c + $(CC) -c -o $@ $< $(CFLAGS) + +clean: + rm -f $(TARGET) $(OBJECTS) + +.PHONY: clean diff --git a/mednafen/libretro/README.md b/mednafen/libretro/README.md new file mode 100644 index 0000000000..1c922d5dd3 --- /dev/null +++ b/mednafen/libretro/README.md @@ -0,0 +1,24 @@ +# Mednafen PSX libretro + +This is port of Mednafen PSX core to the libretro API. +It currently runs on Linux, OSX and possibly Windows. + +## Running + +To run this core, the "system directory" must be defined if running in RetroArch. +Here, the PSX BIOSes must be placed, $sysdir/SCPH550{0,1,2} for Japanese, NA and EU regions respectively. +Memory cards will also be saved to this system directory. + +## Loading ISOs + +Mednafen differs from other PS1 emulators in that it reads a .cue sheet that points to an .iso/.bin whatever. +If you have e.g. foo.iso, you should create a foo.cue, and fill this in: + + FILE "foo.iso" BINARY + TRACK 01 MODE1/2352 + INDEX 01 00:00:00 + +After that, you can load the foo.cue file as a ROM. +Note that this is a dirty hack and will not work on all games. +Ideally, make sure to use rips that have cue-sheets. + diff --git a/mednafen/libretro/config.h b/mednafen/libretro/config.h index 7a332d56da..326b54bbcc 100644 --- a/mednafen/libretro/config.h +++ b/mednafen/libretro/config.h @@ -1,22 +1,34 @@ //win32 msvc config #pragma once +#ifndef __LIBRETRO_CONFIG_H +#define __LIBRETRO_CONFIG_H + #define HAVE__MKDIR 1 + +#if defined(_MSC_VER) && !defined(XBOX360) #define LSB_FIRST +#endif + #define SIZEOF_DOUBLE 8 -#define WANT_DEBUGGER 1 -#define snprintf _snprintf -#define strcasecmp(x,y) _stricmp(x,y) -#define strncasecmp(x, y, l) strnicmp(x, y, l) +#define WANT_DEBUGGER 0 + #define _(x) (x) #define PSS "/" #define round(x) (floorf((x) + 0.5f)) + +#ifdef _MSC_VER #define strdup _strdup #define strtoull _strtoui64 #define strtoll _strtoi64 +#define snprintf _snprintf +#define strcasecmp(x,y) _stricmp(x,y) +#define strncasecmp(x, y, l) strnicmp(x, y, l) +#endif + #define _USE_MATH_DEFINES #define world_strtod strtod -#define WANT_PSX_EMU - #define MEDNAFEN_VERSION "0.999.999-WIP" #define MEDNAFEN_VERSION_NUMERIC 0x999999 + +#endif /* __LIBRETRO_CONFIG_H diff --git a/mednafen/libretro/inttypes.h b/mednafen/libretro/inttypes.h index 6835ef29ab..61e3fd374b 100644 --- a/mednafen/libretro/inttypes.h +++ b/mednafen/libretro/inttypes.h @@ -1,12 +1,7 @@ #pragma once +#ifndef _MEDNAFEN_INTTYPES_H +#define _MEDNAFEN_INTTYPES_H +#include -typedef __int8 int8_t; -typedef __int16 int16_t; -typedef __int32 int32_t; -typedef __int64 int64_t; - -typedef unsigned __int8 uint8_t; -typedef unsigned __int16 uint16_t; -typedef unsigned __int32 uint32_t; -typedef unsigned __int64 uint64_t; +#endif /* MEDNAFEN_INTTYPES_H */ diff --git a/mednafen/libretro/libretro.cpp b/mednafen/libretro/libretro.cpp new file mode 100644 index 0000000000..37cc043f95 --- /dev/null +++ b/mednafen/libretro/libretro.cpp @@ -0,0 +1,581 @@ +#include "mednafen-types.h" +#include "mednafen.h" +#include "git.h" +#include "general.h" +#include "libretro.h" + +static MDFNGI *game; +static retro_video_refresh_t video_cb; +static retro_audio_sample_t audio_cb; +static retro_audio_sample_batch_t audio_batch_cb; +static retro_environment_t environ_cb; +static retro_input_poll_t input_poll_cb; +static retro_input_state_t input_state_cb; + +static MDFN_Surface *surf; + +static bool failed_init; + +std::string retro_base_directory; +std::string retro_base_name; + +#if defined(WANT_PSX_EMU) +#define MEDNAFEN_CORE_NAME_MODULE "psx" +#define MEDNAFEN_CORE_NAME "Mednafen PSX" +#define MEDNAFEN_CORE_VERSION "v0.9.26" +#define MEDNAFEN_CORE_EXTENSIONS "cue|CUE|toc|TOC" +#define MEDNAFEN_CORE_TIMING_FPS 59.85398 +#define MEDNAFEN_CORE_GEOMETRY_BASE_W 320 +#define MEDNAFEN_CORE_GEOMETRY_BASE_H 240 +#define MEDNAFEN_CORE_GEOMETRY_MAX_W 640 +#define MEDNAFEN_CORE_GEOMETRY_MAX_H 480 +#define MEDNAFEN_CORE_GEOMETRY_ASPECT_RATIO (4.0 / 3.0) +#define FB_WIDTH 680 +#define FB_HEIGHT 576 + +#elif defined(WANT_PCE_FAST_EMU) +#define MEDNAFEN_CORE_NAME_MODULE "pce_fast" +#define MEDNAFEN_CORE_NAME "Mednafen PCE Fast" +#define MEDNAFEN_CORE_VERSION "v0.9.24" +#define MEDNAFEN_CORE_EXTENSIONS "pce|PCE|cue|CUE|zip|ZIP" +#define MEDNAFEN_CORE_TIMING_FPS 59.82 +#define MEDNAFEN_CORE_GEOMETRY_BASE_W (game->nominal_width) +#define MEDNAFEN_CORE_GEOMETRY_BASE_H (game->nominal_height) +#define MEDNAFEN_CORE_GEOMETRY_MAX_W 512 +#define MEDNAFEN_CORE_GEOMETRY_MAX_H 242 +#define MEDNAFEN_CORE_GEOMETRY_ASPECT_RATIO (4.0 / 3.0) +#define FB_WIDTH 512 +#define FB_HEIGHT 242 + +#elif defined(WANT_WSWAN_EMU) +#define MEDNAFEN_CORE_NAME_MODULE "wswan" +#define MEDNAFEN_CORE_NAME "Mednafen WonderSwan" +#define MEDNAFEN_CORE_VERSION "v0.9.22" +#define MEDNAFEN_CORE_EXTENSIONS "ws|WS|wsc|WSC|zip|ZIP" +#define MEDNAFEN_CORE_TIMING_FPS 75.47 +#define MEDNAFEN_CORE_GEOMETRY_BASE_W (game->nominal_width) +#define MEDNAFEN_CORE_GEOMETRY_BASE_H (game->nominal_height) +#define MEDNAFEN_CORE_GEOMETRY_MAX_W 224 +#define MEDNAFEN_CORE_GEOMETRY_MAX_H 144 +#define MEDNAFEN_CORE_GEOMETRY_ASPECT_RATIO (4.0 / 3.0) +#define FB_WIDTH 224 +#define FB_HEIGHT 144 + +#endif + +#ifdef WANT_16BPP +static uint16_t mednafen_buf[FB_WIDTH * FB_HEIGHT]; +#else +static uint32_t mednafen_buf[FB_WIDTH * FB_HEIGHT]; +#endif +const char *mednafen_core_str = MEDNAFEN_CORE_NAME; + +static void check_system_specs(void) +{ + unsigned level = 0; +#if defined(WANT_PSX_EMU) + // Hints that we need a fairly powerful system to run this. + level = 3; +#endif + environ_cb(RETRO_ENVIRONMENT_SET_PERFORMANCE_LEVEL, &level); +} + +void retro_init() +{ + std::vector ext; + MDFNI_InitializeModules(ext); + + const char *dir = NULL; + + std::vector settings; + + if (environ_cb(RETRO_ENVIRONMENT_GET_SYSTEM_DIRECTORY, &dir) && dir) + { + retro_base_directory = dir; + // Make sure that we don't have any lingering slashes, etc, as they break Windows. + size_t last = retro_base_directory.find_last_not_of("/\\"); + if (last != std::string::npos) + last++; + + retro_base_directory = retro_base_directory.substr(0, last); + + MDFNI_SetBaseDirectory(retro_base_directory.c_str()); + } + else + { + /* TODO: Add proper fallback */ + fprintf(stderr, "System directory is not defined. Fallback on using same dir as ROM for system directory later ...\n"); + failed_init = true; + } + +#if defined(WANT_16BPP) && defined(FRONTEND_SUPPORTS_RGB565) + enum retro_pixel_format rgb565 = RETRO_PIXEL_FORMAT_RGB565; + if(environ_cb(RETRO_ENVIRONMENT_SET_PIXEL_FORMAT, &rgb565)) + fprintf(stderr, "Frontend supports RGB565 - will use that instead of XRGB1555.\n"); +#endif + + check_system_specs(); +} + +void retro_reset() +{ + MDFNI_Reset(); +} + +bool retro_load_game_special(unsigned, const struct retro_game_info *, size_t) +{ + return false; +} + +bool retro_load_game(const struct retro_game_info *info) +{ + if (failed_init) + return false; + +#ifdef WANT_32BPP + enum retro_pixel_format fmt = RETRO_PIXEL_FORMAT_XRGB8888; + if (!environ_cb(RETRO_ENVIRONMENT_SET_PIXEL_FORMAT, &fmt)) + { + fprintf(stderr, "Pixel format XRGB8888 not supported by platform, cannot use %s.\n", MEDNAFEN_CORE_NAME); + return false; + } +#endif + + const char *base = strrchr(info->path, '/'); + if (!base) + base = strrchr(info->path, '\\'); + + if (base) + retro_base_name = base + 1; + else + retro_base_name = info->path; + + retro_base_name = retro_base_name.substr(0, retro_base_name.find_last_of('.')); + + game = MDFNI_LoadGame(MEDNAFEN_CORE_NAME_MODULE, info->path); + if (!game) + return false; + + MDFN_PixelFormat pix_fmt(MDFN_COLORSPACE_RGB, 16, 8, 0, 24); + + surf = new MDFN_Surface(mednafen_buf, FB_WIDTH, FB_HEIGHT, FB_WIDTH, pix_fmt); + + return game; +} + +void retro_unload_game() +{ + MDFNI_CloseGame(); +} + +static unsigned retro_devices[2]; + +// Hardcoded for PSX. No reason to parse lots of structures ... +// See mednafen/psx/input/gamepad.cpp +static void update_input(void) +{ +#if defined(WANT_PSX_EMU) + union + { + uint32_t u32[2][1 + 8]; + uint8_t u8[2][2 * sizeof(uint16_t) + 8 * sizeof(uint32_t)]; + } static buf; + + uint16_t input_buf[2] = {0}; + static unsigned map[] = { + RETRO_DEVICE_ID_JOYPAD_SELECT, + RETRO_DEVICE_ID_JOYPAD_L3, + RETRO_DEVICE_ID_JOYPAD_R3, + RETRO_DEVICE_ID_JOYPAD_START, + RETRO_DEVICE_ID_JOYPAD_UP, + RETRO_DEVICE_ID_JOYPAD_RIGHT, + RETRO_DEVICE_ID_JOYPAD_DOWN, + RETRO_DEVICE_ID_JOYPAD_LEFT, + RETRO_DEVICE_ID_JOYPAD_L2, + RETRO_DEVICE_ID_JOYPAD_R2, + RETRO_DEVICE_ID_JOYPAD_L, + RETRO_DEVICE_ID_JOYPAD_R, + RETRO_DEVICE_ID_JOYPAD_X, + RETRO_DEVICE_ID_JOYPAD_A, + RETRO_DEVICE_ID_JOYPAD_B, + RETRO_DEVICE_ID_JOYPAD_Y, + }; + + for (unsigned j = 0; j < 2; j++) + { + for (unsigned i = 0; i < 16; i++) + input_buf[j] |= input_state_cb(j, RETRO_DEVICE_JOYPAD, 0, map[i]) ? (1 << i) : 0; + } + + // Buttons. + buf.u8[0][0] = (input_buf[0] >> 0) & 0xff; + buf.u8[0][1] = (input_buf[0] >> 8) & 0xff; + buf.u8[1][0] = (input_buf[1] >> 0) & 0xff; + buf.u8[1][1] = (input_buf[1] >> 8) & 0xff; + + // Analogs + for (unsigned j = 0; j < 2; j++) + { + int analog_left_x = input_state_cb(j, RETRO_DEVICE_ANALOG, RETRO_DEVICE_INDEX_ANALOG_LEFT, + RETRO_DEVICE_ID_ANALOG_X); + + int analog_left_y = input_state_cb(j, RETRO_DEVICE_ANALOG, RETRO_DEVICE_INDEX_ANALOG_LEFT, + RETRO_DEVICE_ID_ANALOG_Y); + + int analog_right_x = input_state_cb(j, RETRO_DEVICE_ANALOG, RETRO_DEVICE_INDEX_ANALOG_RIGHT, + RETRO_DEVICE_ID_ANALOG_X); + + int analog_right_y = input_state_cb(j, RETRO_DEVICE_ANALOG, RETRO_DEVICE_INDEX_ANALOG_RIGHT, + RETRO_DEVICE_ID_ANALOG_Y); + + uint32_t r_right = analog_right_x > 0 ? analog_right_x : 0; + uint32_t r_left = analog_right_x < 0 ? -analog_right_x : 0; + uint32_t r_down = analog_right_y > 0 ? analog_right_y : 0; + uint32_t r_up = analog_right_y < 0 ? -analog_right_y : 0; + + uint32_t l_right = analog_left_x > 0 ? analog_left_x : 0; + uint32_t l_left = analog_left_x < 0 ? -analog_left_x : 0; + uint32_t l_down = analog_left_y > 0 ? analog_left_y : 0; + uint32_t l_up = analog_left_y < 0 ? -analog_left_y : 0; + + buf.u32[j][1] = r_right; + buf.u32[j][2] = r_left; + buf.u32[j][3] = r_down; + buf.u32[j][4] = r_up; + + buf.u32[j][5] = l_right; + buf.u32[j][6] = l_left; + buf.u32[j][7] = l_down; + buf.u32[j][8] = l_up; + } + + for (int j = 0; j < 2; j++) + { + switch (retro_devices[j]) + { + case RETRO_DEVICE_ANALOG: + game->SetInput(j, "dualanalog", &buf.u8[j]); + break; + default: + game->SetInput(j, "gamepad", &buf.u8[j]); + break; + } + } +#elif defined(WANT_PCE_FAST_EMU) + static uint8_t input_buf[5][2]; + + static unsigned map[] = { + RETRO_DEVICE_ID_JOYPAD_Y, + RETRO_DEVICE_ID_JOYPAD_B, + RETRO_DEVICE_ID_JOYPAD_SELECT, + RETRO_DEVICE_ID_JOYPAD_START, + RETRO_DEVICE_ID_JOYPAD_UP, + RETRO_DEVICE_ID_JOYPAD_RIGHT, + RETRO_DEVICE_ID_JOYPAD_DOWN, + RETRO_DEVICE_ID_JOYPAD_LEFT, + RETRO_DEVICE_ID_JOYPAD_A, + RETRO_DEVICE_ID_JOYPAD_X, + RETRO_DEVICE_ID_JOYPAD_L, + RETRO_DEVICE_ID_JOYPAD_R, + RETRO_DEVICE_ID_JOYPAD_L2 + }; + + if (input_state_cb) + { + for (unsigned j = 0; j < 5; j++) + { + uint16_t input_state = 0; + for (unsigned i = 0; i < 13; i++) + input_state |= input_state_cb(j, RETRO_DEVICE_JOYPAD, 0, map[i]) ? (1 << i) : 0; + + // Input data must be little endian. + input_buf[j][0] = (input_state >> 0) & 0xff; + input_buf[j][1] = (input_state >> 8) & 0xff; + } + } + + // Possible endian bug ... + for (unsigned i = 0; i < 5; i++) + MDFNI_SetInput(i, "gamepad", &input_buf[i][0], 0); +#elif defined(WANT_WSWAN_EMU) + static uint16_t input_buf; + input_buf = 0; + + static unsigned map[] = { + RETRO_DEVICE_ID_JOYPAD_UP, //X Cursor horizontal-layout games + RETRO_DEVICE_ID_JOYPAD_RIGHT, //X Cursor horizontal-layout games + RETRO_DEVICE_ID_JOYPAD_DOWN, //X Cursor horizontal-layout games + RETRO_DEVICE_ID_JOYPAD_LEFT, //X Cursor horizontal-layout games + RETRO_DEVICE_ID_JOYPAD_R2, //Y Cursor UP vertical-layout games + RETRO_DEVICE_ID_JOYPAD_R, //Y Cursor RIGHT vertical-layout games + RETRO_DEVICE_ID_JOYPAD_L2, //Y Cursor DOWN vertical-layout games + RETRO_DEVICE_ID_JOYPAD_L, //Y Cursor LEFT vertical-layout games + RETRO_DEVICE_ID_JOYPAD_START, + RETRO_DEVICE_ID_JOYPAD_A, + RETRO_DEVICE_ID_JOYPAD_B, + }; + +for (unsigned i = 0; i < 11; i++) + input_buf |= map[i] != -1u && + input_state_cb(0, RETRO_DEVICE_JOYPAD, 0, map[i]) ? (1 << i) : 0; + +#ifdef MSB_FIRST + union { + uint8_t b[2]; + uint16_t s; + } u; + u.s = input_buf; + input_buf = u.b[0] | u.b[1] << 8; +#endif + + game->SetInput(0, "gamepad", &input_buf); +#endif +} + +static uint64_t video_frames, audio_frames; + +void retro_run() +{ + input_poll_cb(); + + update_input(); + + static int16_t sound_buf[0x10000]; +#if defined(WANT_WSWAN_EMU) + static MDFN_Rect rects[FB_WIDTH]; +#else + static MDFN_Rect rects[FB_HEIGHT]; +#endif +#if defined(WANT_PSX_EMU) || defined(WANT_WSWAN_EMU) + rects[0].w = ~0; +#endif + + EmulateSpecStruct spec = {0}; + spec.surface = surf; + spec.SoundRate = 44100; + spec.SoundBuf = sound_buf; + spec.LineWidths = rects; + spec.SoundBufMaxSize = sizeof(sound_buf) / 2; + spec.SoundVolume = 1.0; + spec.soundmultiplier = 1.0; + + MDFNI_Emulate(&spec); + +#if defined(WANT_PSX_EMU) + unsigned width = rects[0].w; + unsigned height = spec.DisplayRect.h; + unsigned int ptrDiff = 0; +#elif defined(WANT_WSWAN_EMU) + unsigned width = FB_WIDTH; + unsigned height = FB_HEIGHT; +#else + unsigned width = rects->w; + unsigned height = rects->h; +#endif + +#ifdef WANT_PSX_EMU + // This is for PAL, the core implements PAL over NTSC TV so you get the + // infamous PAL borders. This removes them. The PS1 supports only two horizontal + // resolutions so it's OK to use constants and not percentage. + bool isPal = false; + if (height == FB_HEIGHT) + { + ptrDiff += width * 47; + height = 480; + isPal = true; + } + else if (height == 288) + { + // TODO: This seems to be OK as is, but I might be wrong. + isPal = true; + } + + if (isPal && width == FB_WIDTH) + ptrDiff += 7; + + // The core handles vertical overscan for NTSC pretty well, but it ignores + // horizontal overscan. This is a tough estimation of what the horizontal + // overscan should be, tested with all major NTSC resolutions. Mayeb make it + // configurable? + float hoverscan = 0.941176471; + + width = width * hoverscan; + ptrDiff += ((rects[0].w - width) / 2); + + const uint32_t *ptr = surf->pixels; + ptr += ptrDiff; + + video_cb(ptr, width, height, FB_WIDTH << 2); +#elif defined(WANT_16BPP) + const uint16_t *pix = surf->pixels16; + video_cb(pix, width, height, FB_WIDTH << 1); +#endif + + video_frames++; + audio_frames += spec.SoundBufSize; + + audio_batch_cb(spec.SoundBuf, spec.SoundBufSize); +} + +void retro_get_system_info(struct retro_system_info *info) +{ + memset(info, 0, sizeof(*info)); + info->library_name = MEDNAFEN_CORE_NAME; + info->library_version = MEDNAFEN_CORE_VERSION; + info->need_fullpath = true; + info->valid_extensions = MEDNAFEN_CORE_EXTENSIONS; + info->block_extract = false; +} + +void retro_get_system_av_info(struct retro_system_av_info *info) +{ + memset(info, 0, sizeof(*info)); + info->timing.fps = MEDNAFEN_CORE_TIMING_FPS; // Determined from empirical testing. + info->timing.sample_rate = 44100; + info->geometry.base_width = MEDNAFEN_CORE_GEOMETRY_BASE_W; + info->geometry.base_height = MEDNAFEN_CORE_GEOMETRY_BASE_H; + info->geometry.max_width = MEDNAFEN_CORE_GEOMETRY_MAX_W; + info->geometry.max_height = MEDNAFEN_CORE_GEOMETRY_MAX_H; + info->geometry.aspect_ratio = MEDNAFEN_CORE_GEOMETRY_ASPECT_RATIO; +} + +void retro_deinit() +{ + delete surf; + surf = NULL; + + fprintf(stderr, "[%s]: Samples / Frame: %.5f\n", + mednafen_core_str, (double)audio_frames / video_frames); + + fprintf(stderr, "[%s]: Estimated FPS: %.5f\n", + mednafen_core_str, (double)video_frames * 44100 / audio_frames); +} + +unsigned retro_get_region(void) +{ + return RETRO_REGION_NTSC; +} + +unsigned retro_api_version(void) +{ + return RETRO_API_VERSION; +} + +void retro_set_controller_port_device(unsigned in_port, unsigned device) +{ +#ifdef WANT_PSX_EMU + if (in_port > 1) + { + fprintf(stderr, + "[%s]: Only the 2 main ports are supported at the moment", mednafen_core_str); + return; + } + + switch (device) + { + // TODO: Add support for other input types + case RETRO_DEVICE_JOYPAD: + case RETRO_DEVICE_ANALOG: + fprintf(stderr, "[%s]: Selected controller type %u", mednafen_core_str, device); + retro_devices[in_port] = device; + break; + default: + retro_devices[in_port] = RETRO_DEVICE_JOYPAD; + fprintf(stderr, + "[%s]: Unsupported controller device, falling back to gamepad", mednafen_core_str); + } +#endif +} + +void retro_set_environment(retro_environment_t cb) +{ + environ_cb = cb; +} + +void retro_set_audio_sample(retro_audio_sample_t cb) +{ + audio_cb = cb; +} + +void retro_set_audio_sample_batch(retro_audio_sample_batch_t cb) +{ + audio_batch_cb = cb; +} + +void retro_set_input_poll(retro_input_poll_t cb) +{ + input_poll_cb = cb; +} + +void retro_set_input_state(retro_input_state_t cb) +{ + input_state_cb = cb; +} + +void retro_set_video_refresh(retro_video_refresh_t cb) +{ + video_cb = cb; +} + +static size_t serialize_size; + +size_t retro_serialize_size(void) +{ + //if (serialize_size) + // return serialize_size; + + if (!game->StateAction) + { + fprintf(stderr, "[mednafen]: Module %s doesn't support save states.\n", game->shortname); + return 0; + } + + StateMem st; + memset(&st, 0, sizeof(st)); + + if (!MDFNSS_SaveSM(&st, 0, 0, NULL, NULL, NULL)) + { + fprintf(stderr, "[mednafen]: Module %s doesn't support save states.\n", game->shortname); + return 0; + } + + free(st.data); + return serialize_size = st.len; +} + +bool retro_serialize(void *data, size_t size) +{ + StateMem st; + memset(&st, 0, sizeof(st)); + st.data = (uint8_t*)data; + st.malloced = size; + + return MDFNSS_SaveSM(&st, 0, 0, NULL, NULL, NULL); +} + +bool retro_unserialize(const void *data, size_t size) +{ + StateMem st; + memset(&st, 0, sizeof(st)); + st.data = (uint8_t*)data; + st.len = size; + + return MDFNSS_LoadSM(&st, 0, 0); +} + +void *retro_get_memory_data(unsigned) +{ + return NULL; +} + +size_t retro_get_memory_size(unsigned) +{ + return 0; +} + +void retro_cheat_reset(void) +{} + +void retro_cheat_set(unsigned, bool, const char *) +{} + diff --git a/mednafen/libretro/libretro.h b/mednafen/libretro/libretro.h new file mode 100755 index 0000000000..3eef758841 --- /dev/null +++ b/mednafen/libretro/libretro.h @@ -0,0 +1,577 @@ +#ifndef LIBRETRO_H__ +#define LIBRETRO_H__ + +#include +#include +#include + +// Hack applied for MSVC when compiling in C89 mode as it isn't C99 compliant. +#ifdef __cplusplus +extern "C" { +#else +#if defined(_MSC_VER) && !defined(SN_TARGET_PS3) && !defined(__cplusplus) +#define bool unsigned char +#define true 1 +#define false 0 +#else +#include +#endif +#endif + +// Used for checking API/ABI mismatches that can break libretro implementations. +// It is not incremented for compatible changes. +#define RETRO_API_VERSION 1 + +// Libretro's fundamental device abstractions. +#define RETRO_DEVICE_MASK 0xff +#define RETRO_DEVICE_NONE 0 + +// The JOYPAD is called RetroPad. It is essentially a Super Nintendo controller, +// but with additional L2/R2/L3/R3 buttons, similar to a PS1 DualShock. +#define RETRO_DEVICE_JOYPAD 1 + +// The mouse is a simple mouse, similar to Super Nintendo's mouse. +// X and Y coordinates are reported relatively to last poll (poll callback). +// It is up to the libretro implementation to keep track of where the mouse pointer is supposed to be on the screen. +// The frontend must make sure not to interfere with its own hardware mouse pointer. +#define RETRO_DEVICE_MOUSE 2 + +// KEYBOARD device lets one poll for raw key pressed. +// It is poll based, so input callback will return with the current pressed state. +#define RETRO_DEVICE_KEYBOARD 3 + +// Lightgun X/Y coordinates are reported relatively to last poll, similar to mouse. +#define RETRO_DEVICE_LIGHTGUN 4 + +// The ANALOG device is an extension to JOYPAD (RetroPad). +// Similar to DualShock it adds two analog sticks. +// This is treated as a separate device type as it returns values in the full analog range +// of [-0x8000, 0x7fff]. Positive X axis is right. Positive Y axis is down. +// Only use ANALOG type when polling for analog values of the axes. +#define RETRO_DEVICE_ANALOG 5 + +// These device types are specializations of the base types above. +// They should only be used in retro_set_controller_type() to inform libretro implementations +// about use of a very specific device type. +// +// In input state callback, however, only the base type should be used in the 'device' field. +#define RETRO_DEVICE_JOYPAD_MULTITAP ((1 << 8) | RETRO_DEVICE_JOYPAD) +#define RETRO_DEVICE_LIGHTGUN_SUPER_SCOPE ((1 << 8) | RETRO_DEVICE_LIGHTGUN) +#define RETRO_DEVICE_LIGHTGUN_JUSTIFIER ((2 << 8) | RETRO_DEVICE_LIGHTGUN) +#define RETRO_DEVICE_LIGHTGUN_JUSTIFIERS ((3 << 8) | RETRO_DEVICE_LIGHTGUN) + +// Buttons for the RetroPad (JOYPAD). +// The placement of these is equivalent to placements on the Super Nintendo controller. +// L2/R2/L3/R3 buttons correspond to the PS1 DualShock. +#define RETRO_DEVICE_ID_JOYPAD_B 0 +#define RETRO_DEVICE_ID_JOYPAD_Y 1 +#define RETRO_DEVICE_ID_JOYPAD_SELECT 2 +#define RETRO_DEVICE_ID_JOYPAD_START 3 +#define RETRO_DEVICE_ID_JOYPAD_UP 4 +#define RETRO_DEVICE_ID_JOYPAD_DOWN 5 +#define RETRO_DEVICE_ID_JOYPAD_LEFT 6 +#define RETRO_DEVICE_ID_JOYPAD_RIGHT 7 +#define RETRO_DEVICE_ID_JOYPAD_A 8 +#define RETRO_DEVICE_ID_JOYPAD_X 9 +#define RETRO_DEVICE_ID_JOYPAD_L 10 +#define RETRO_DEVICE_ID_JOYPAD_R 11 +#define RETRO_DEVICE_ID_JOYPAD_L2 12 +#define RETRO_DEVICE_ID_JOYPAD_R2 13 +#define RETRO_DEVICE_ID_JOYPAD_L3 14 +#define RETRO_DEVICE_ID_JOYPAD_R3 15 + +// Index / Id values for ANALOG device. +#define RETRO_DEVICE_INDEX_ANALOG_LEFT 0 +#define RETRO_DEVICE_INDEX_ANALOG_RIGHT 1 +#define RETRO_DEVICE_ID_ANALOG_X 0 +#define RETRO_DEVICE_ID_ANALOG_Y 1 + +// Id values for MOUSE. +#define RETRO_DEVICE_ID_MOUSE_X 0 +#define RETRO_DEVICE_ID_MOUSE_Y 1 +#define RETRO_DEVICE_ID_MOUSE_LEFT 2 +#define RETRO_DEVICE_ID_MOUSE_RIGHT 3 + +// Id values for LIGHTGUN types. +#define RETRO_DEVICE_ID_LIGHTGUN_X 0 +#define RETRO_DEVICE_ID_LIGHTGUN_Y 1 +#define RETRO_DEVICE_ID_LIGHTGUN_TRIGGER 2 +#define RETRO_DEVICE_ID_LIGHTGUN_CURSOR 3 +#define RETRO_DEVICE_ID_LIGHTGUN_TURBO 4 +#define RETRO_DEVICE_ID_LIGHTGUN_PAUSE 5 +#define RETRO_DEVICE_ID_LIGHTGUN_START 6 + +// Returned from retro_get_region(). +#define RETRO_REGION_NTSC 0 +#define RETRO_REGION_PAL 1 + +// Passed to retro_get_memory_data/size(). +// If the memory type doesn't apply to the implementation NULL/0 can be returned. +#define RETRO_MEMORY_MASK 0xff + +// Regular save ram. This ram is usually found on a game cartridge, backed up by a battery. +// If save game data is too complex for a single memory buffer, +// the SYSTEM_DIRECTORY environment callback can be used. +#define RETRO_MEMORY_SAVE_RAM 0 + +// Some games have a built-in clock to keep track of time. +// This memory is usually just a couple of bytes to keep track of time. +#define RETRO_MEMORY_RTC 1 + +// System ram lets a frontend peek into a game systems main RAM. +#define RETRO_MEMORY_SYSTEM_RAM 2 + +// Video ram lets a frontend peek into a game systems video RAM (VRAM). +#define RETRO_MEMORY_VIDEO_RAM 3 + +// Special memory types. +#define RETRO_MEMORY_SNES_BSX_RAM ((1 << 8) | RETRO_MEMORY_SAVE_RAM) +#define RETRO_MEMORY_SNES_BSX_PRAM ((2 << 8) | RETRO_MEMORY_SAVE_RAM) +#define RETRO_MEMORY_SNES_SUFAMI_TURBO_A_RAM ((3 << 8) | RETRO_MEMORY_SAVE_RAM) +#define RETRO_MEMORY_SNES_SUFAMI_TURBO_B_RAM ((4 << 8) | RETRO_MEMORY_SAVE_RAM) +#define RETRO_MEMORY_SNES_GAME_BOY_RAM ((5 << 8) | RETRO_MEMORY_SAVE_RAM) +#define RETRO_MEMORY_SNES_GAME_BOY_RTC ((6 << 8) | RETRO_MEMORY_RTC) + +// Special game types passed into retro_load_game_special(). +// Only used when multiple ROMs are required. +#define RETRO_GAME_TYPE_BSX 0x101 +#define RETRO_GAME_TYPE_BSX_SLOTTED 0x102 +#define RETRO_GAME_TYPE_SUFAMI_TURBO 0x103 +#define RETRO_GAME_TYPE_SUPER_GAME_BOY 0x104 + +// Keysyms used for ID in input state callback when polling RETRO_KEYBOARD. +enum retro_key +{ + RETROK_UNKNOWN = 0, + RETROK_FIRST = 0, + RETROK_BACKSPACE = 8, + RETROK_TAB = 9, + RETROK_CLEAR = 12, + RETROK_RETURN = 13, + RETROK_PAUSE = 19, + RETROK_ESCAPE = 27, + RETROK_SPACE = 32, + RETROK_EXCLAIM = 33, + RETROK_QUOTEDBL = 34, + RETROK_HASH = 35, + RETROK_DOLLAR = 36, + RETROK_AMPERSAND = 38, + RETROK_QUOTE = 39, + RETROK_LEFTPAREN = 40, + RETROK_RIGHTPAREN = 41, + RETROK_ASTERISK = 42, + RETROK_PLUS = 43, + RETROK_COMMA = 44, + RETROK_MINUS = 45, + RETROK_PERIOD = 46, + RETROK_SLASH = 47, + RETROK_0 = 48, + RETROK_1 = 49, + RETROK_2 = 50, + RETROK_3 = 51, + RETROK_4 = 52, + RETROK_5 = 53, + RETROK_6 = 54, + RETROK_7 = 55, + RETROK_8 = 56, + RETROK_9 = 57, + RETROK_COLON = 58, + RETROK_SEMICOLON = 59, + RETROK_LESS = 60, + RETROK_EQUALS = 61, + RETROK_GREATER = 62, + RETROK_QUESTION = 63, + RETROK_AT = 64, + RETROK_LEFTBRACKET = 91, + RETROK_BACKSLASH = 92, + RETROK_RIGHTBRACKET = 93, + RETROK_CARET = 94, + RETROK_UNDERSCORE = 95, + RETROK_BACKQUOTE = 96, + RETROK_a = 97, + RETROK_b = 98, + RETROK_c = 99, + RETROK_d = 100, + RETROK_e = 101, + RETROK_f = 102, + RETROK_g = 103, + RETROK_h = 104, + RETROK_i = 105, + RETROK_j = 106, + RETROK_k = 107, + RETROK_l = 108, + RETROK_m = 109, + RETROK_n = 110, + RETROK_o = 111, + RETROK_p = 112, + RETROK_q = 113, + RETROK_r = 114, + RETROK_s = 115, + RETROK_t = 116, + RETROK_u = 117, + RETROK_v = 118, + RETROK_w = 119, + RETROK_x = 120, + RETROK_y = 121, + RETROK_z = 122, + RETROK_DELETE = 127, + + RETROK_KP0 = 256, + RETROK_KP1 = 257, + RETROK_KP2 = 258, + RETROK_KP3 = 259, + RETROK_KP4 = 260, + RETROK_KP5 = 261, + RETROK_KP6 = 262, + RETROK_KP7 = 263, + RETROK_KP8 = 264, + RETROK_KP9 = 265, + RETROK_KP_PERIOD = 266, + RETROK_KP_DIVIDE = 267, + RETROK_KP_MULTIPLY = 268, + RETROK_KP_MINUS = 269, + RETROK_KP_PLUS = 270, + RETROK_KP_ENTER = 271, + RETROK_KP_EQUALS = 272, + + RETROK_UP = 273, + RETROK_DOWN = 274, + RETROK_RIGHT = 275, + RETROK_LEFT = 276, + RETROK_INSERT = 277, + RETROK_HOME = 278, + RETROK_END = 279, + RETROK_PAGEUP = 280, + RETROK_PAGEDOWN = 281, + + RETROK_F1 = 282, + RETROK_F2 = 283, + RETROK_F3 = 284, + RETROK_F4 = 285, + RETROK_F5 = 286, + RETROK_F6 = 287, + RETROK_F7 = 288, + RETROK_F8 = 289, + RETROK_F9 = 290, + RETROK_F10 = 291, + RETROK_F11 = 292, + RETROK_F12 = 293, + RETROK_F13 = 294, + RETROK_F14 = 295, + RETROK_F15 = 296, + + RETROK_NUMLOCK = 300, + RETROK_CAPSLOCK = 301, + RETROK_SCROLLOCK = 302, + RETROK_RSHIFT = 303, + RETROK_LSHIFT = 304, + RETROK_RCTRL = 305, + RETROK_LCTRL = 306, + RETROK_RALT = 307, + RETROK_LALT = 308, + RETROK_RMETA = 309, + RETROK_LMETA = 310, + RETROK_LSUPER = 311, + RETROK_RSUPER = 312, + RETROK_MODE = 313, + RETROK_COMPOSE = 314, + + RETROK_HELP = 315, + RETROK_PRINT = 316, + RETROK_SYSREQ = 317, + RETROK_BREAK = 318, + RETROK_MENU = 319, + RETROK_POWER = 320, + RETROK_EURO = 321, + RETROK_UNDO = 322, + + RETROK_LAST +}; + +// Environment commands. +#define RETRO_ENVIRONMENT_SET_ROTATION 1 // const unsigned * -- + // Sets screen rotation of graphics. + // Is only implemented if rotation can be accelerated by hardware. + // Valid values are 0, 1, 2, 3, which rotates screen by 0, 90, 180, 270 degrees + // counter-clockwise respectively. + // +#define RETRO_ENVIRONMENT_GET_OVERSCAN 2 // bool * -- + // Boolean value whether or not the implementation should use overscan, or crop away overscan. + // +#define RETRO_ENVIRONMENT_GET_CAN_DUPE 3 // bool * -- + // Boolean value whether or not frontend supports frame duping, + // passing NULL to video frame callback. + // +#define RETRO_ENVIRONMENT_GET_VARIABLE 4 // struct retro_variable * -- + // Interface to aquire user-defined information from environment + // that cannot feasibly be supported in a multi-system way. + // Mostly used for obscure, + // specific features that the user can tap into when neseccary. + // +#define RETRO_ENVIRONMENT_SET_VARIABLES 5 // const struct retro_variable * -- + // Allows an implementation to signal the environment + // which variables it might want to check for later using GET_VARIABLE. + // 'data' points to an array of retro_variable structs terminated by a { NULL, NULL } element. + // retro_variable::value should contain a human readable description of the key. + // +#define RETRO_ENVIRONMENT_SET_MESSAGE 6 // const struct retro_message * -- + // Sets a message to be displayed in implementation-specific manner for a certain amount of 'frames'. + // Should not be used for trivial messages, which should simply be logged to stderr. +#define RETRO_ENVIRONMENT_SHUTDOWN 7 // N/A (NULL) -- + // Requests the frontend to shutdown. + // Should only be used if game has a specific + // way to shutdown the game from a menu item or similar. + // +#define RETRO_ENVIRONMENT_SET_PERFORMANCE_LEVEL 8 + // const unsigned * -- + // Gives a hint to the frontend how demanding this implementation + // is on a system. E.g. reporting a level of 2 means + // this implementation should run decently on all frontends + // of level 2 and up. + // + // It can be used by the frontend to potentially warn + // about too demanding implementations. + // + // The levels are "floating", but roughly defined as: + // 0: Low-powered embedded devices such as Raspberry Pi + // 1: 6th generation consoles, such as Wii/Xbox 1, and phones, tablets, etc. + // 2: 7th generation consoles, such as PS3/360, with sub-par CPUs. + // 3: Modern desktop/laptops with reasonably powerful CPUs. + // 4: High-end desktops with very powerful CPUs. + // + // This function can be called on a per-game basis, + // as certain games an implementation can play might be + // particularily demanding. + // If called, it should be called in retro_load_game(). + // +#define RETRO_ENVIRONMENT_GET_SYSTEM_DIRECTORY 9 + // const char ** -- + // Returns the "system" directory of the frontend. + // This directory can be used to store system specific ROMs such as BIOSes, configuration data, etc. + // The returned value can be NULL. + // If so, no such directory is defined, + // and it's up to the implementation to find a suitable directory. + // +#define RETRO_ENVIRONMENT_SET_PIXEL_FORMAT 10 + // const enum retro_pixel_format * -- + // Sets the internal pixel format used by the implementation. + // The default pixel format is RETRO_PIXEL_FORMAT_0RGB1555. + // This pixel format however, is deprecated (see enum retro_pixel_format). + // If the call returns false, the frontend does not support this pixel format. + // This function should be called inside retro_load_game() or retro_get_system_av_info(). + // +#define RETRO_ENVIRONMENT_SET_INPUT_DESCRIPTORS 11 + // const struct retro_input_descriptor * -- + // Sets an array of retro_input_descriptors. + // It is up to the frontend to present this in a usable way. + // The array is terminated by retro_input_descriptor::description being set to NULL. + // This function can be called at any time, but it is recommended to call it as early as possible. + + +enum retro_pixel_format +{ + // 0RGB1555, native endian. 0 bit must be set to 0. + // This pixel format is default for compatibility concerns only. + // If a 15/16-bit pixel format is desired, consider using RGB565. + RETRO_PIXEL_FORMAT_0RGB1555 = 0, + + // XRGB8888, native endian. X bits are ignored. + RETRO_PIXEL_FORMAT_XRGB8888 = 1, + + // RGB565, native endian. This pixel format is the recommended format to use if a 15/16-bit format is desired + // as it is the pixel format that is typically available on a wide range of low-power devices. + // It is also natively supported in APIs like OpenGL ES. + RETRO_PIXEL_FORMAT_RGB565 = 2, + + // Ensure sizeof() == sizeof(int). + RETRO_PIXEL_FORMAT_UNKNOWN = INT_MAX +}; + +struct retro_message +{ + const char *msg; // Message to be displayed. + unsigned frames; // Duration in frames of message. +}; + +// Describes how the libretro implementation maps a libretro input bind +// to its internal input system through a human readable string. +// This string can be used to better let a user configure input. +struct retro_input_descriptor +{ + // Associates given parameters with a description. + unsigned port; + unsigned device; + unsigned index; + unsigned id; + + const char *description; // Human readable description for parameters. + // The pointer must remain valid until retro_unload_game() is called. +}; + +struct retro_system_info +{ + // All pointers are owned by libretro implementation, and pointers must remain valid until retro_deinit() is called. + + const char *library_name; // Descriptive name of library. Should not contain any version numbers, etc. + const char *library_version; // Descriptive version of core. + + const char *valid_extensions; // A string listing probably rom extensions the core will be able to load, separated with pipe. + // I.e. "bin|rom|iso". + // Typically used for a GUI to filter out extensions. + + bool need_fullpath; // If true, retro_load_game() is guaranteed to provide a valid pathname in retro_game_info::path. + // ::data and ::size are both invalid. + // If false, ::data and ::size are guaranteed to be valid, but ::path might not be valid. + // This is typically set to true for libretro implementations that must load from file. + // Implementations should strive for setting this to false, as it allows the frontend to perform patching, etc. + + bool block_extract; // If true, the frontend is not allowed to extract any archives before loading the real ROM. + // Necessary for certain libretro implementations that load games from zipped archives. +}; + +struct retro_game_geometry +{ + unsigned base_width; // Nominal video width of game. + unsigned base_height; // Nominal video height of game. + unsigned max_width; // Maximum possible width of game. + unsigned max_height; // Maximum possible height of game. + + float aspect_ratio; // Nominal aspect ratio of game. If aspect_ratio is <= 0.0, + // an aspect ratio of base_width / base_height is assumed. + // A frontend could override this setting if desired. +}; + +struct retro_system_timing +{ + double fps; // FPS of video content. + double sample_rate; // Sampling rate of audio. +}; + +struct retro_system_av_info +{ + struct retro_game_geometry geometry; + struct retro_system_timing timing; +}; + +struct retro_variable +{ + const char *key; // Variable to query in RETRO_ENVIRONMENT_GET_VARIABLE. + // If NULL, obtains the complete environment string if more complex parsing is necessary. + // The environment string is formatted as key-value pairs delimited by semicolons as so: + // "key1=value1;key2=value2;..." + const char *value; // Value to be obtained. If key does not exist, it is set to NULL. +}; + +struct retro_game_info +{ + const char *path; // Path to game, UTF-8 encoded. Usually used as a reference. + // May be NULL if rom was loaded from stdin or similar. + // retro_system_info::need_fullpath guaranteed that this path is valid. + const void *data; // Memory buffer of loaded game. Will be NULL if need_fullpath was set. + size_t size; // Size of memory buffer. + const char *meta; // String of implementation specific meta-data. +}; + +// Callbacks +// +// Environment callback. Gives implementations a way of performing uncommon tasks. Extensible. +typedef bool (*retro_environment_t)(unsigned cmd, void *data); + +// Render a frame. Pixel format is 15-bit 0RGB1555 native endian unless changed (see RETRO_ENVIRONMENT_SET_PIXEL_FORMAT). +// Width and height specify dimensions of buffer. +// Pitch specifices length in bytes between two lines in buffer. +// For performance reasons, it is highly recommended to have a frame that is packed in memory, i.e. pitch == width * byte_per_pixel. +// Certain graphic APIs, such as OpenGL ES, do not like textures that are not packed in memory. +typedef void (*retro_video_refresh_t)(const void *data, unsigned width, unsigned height, size_t pitch); + +// Renders a single audio frame. Should only be used if implementation generates a single sample at a time. +// Format is signed 16-bit native endian. +typedef void (*retro_audio_sample_t)(int16_t left, int16_t right); +// Renders multiple audio frames in one go. One frame is defined as a sample of left and right channels, interleaved. +// I.e. int16_t buf[4] = { l, r, l, r }; would be 2 frames. +// Only one of the audio callbacks must ever be used. +typedef size_t (*retro_audio_sample_batch_t)(const int16_t *data, size_t frames); + +// Polls input. +typedef void (*retro_input_poll_t)(void); +// Queries for input for player 'port'. device will be masked with RETRO_DEVICE_MASK. +// Specialization of devices such as RETRO_DEVICE_JOYPAD_MULTITAP that have been set with retro_set_controller_port_device() +// will still use the higher level RETRO_DEVICE_JOYPAD to request input. +typedef int16_t (*retro_input_state_t)(unsigned port, unsigned device, unsigned index, unsigned id); + +// Sets callbacks. retro_set_environment() is guaranteed to be called before retro_init(). +// The rest of the set_* functions are guaranteed to have been called before the first call to retro_run() is made. +void retro_set_environment(retro_environment_t); +void retro_set_video_refresh(retro_video_refresh_t); +void retro_set_audio_sample(retro_audio_sample_t); +void retro_set_audio_sample_batch(retro_audio_sample_batch_t); +void retro_set_input_poll(retro_input_poll_t); +void retro_set_input_state(retro_input_state_t); + +// Library global initialization/deinitialization. +void retro_init(void); +void retro_deinit(void); + +// Must return RETRO_API_VERSION. Used to validate ABI compatibility when the API is revised. +unsigned retro_api_version(void); + +// Gets statically known system info. Pointers provided in *info must be statically allocated. +// Can be called at any time, even before retro_init(). +void retro_get_system_info(struct retro_system_info *info); + +// Gets information about system audio/video timings and geometry. +// Can be called only after retro_load_game() has successfully completed. +// NOTE: The implementation of this function might not initialize every variable if needed. +// E.g. geom.aspect_ratio might not be initialized if core doesn't desire a particular aspect ratio. +void retro_get_system_av_info(struct retro_system_av_info *info); + +// Sets device to be used for player 'port'. +void retro_set_controller_port_device(unsigned port, unsigned device); + +// Resets the current game. +void retro_reset(void); + +// Runs the game for one video frame. +// During retro_run(), input_poll callback must be called at least once. +// +// If a frame is not rendered for reasons where a game "dropped" a frame, +// this still counts as a frame, and retro_run() should explicitly dupe a frame if GET_CAN_DUPE returns true. +// In this case, the video callback can take a NULL argument for data. +void retro_run(void); + +// Returns the amount of data the implementation requires to serialize internal state (save states). +// Beetween calls to retro_load_game() and retro_unload_game(), the returned size is never allowed to be larger than a previous returned value, to +// ensure that the frontend can allocate a save state buffer once. +size_t retro_serialize_size(void); + +// Serializes internal state. If failed, or size is lower than retro_serialize_size(), it should return false, true otherwise. +bool retro_serialize(void *data, size_t size); +bool retro_unserialize(const void *data, size_t size); + +void retro_cheat_reset(void); +void retro_cheat_set(unsigned index, bool enabled, const char *code); + +// Loads a game. +bool retro_load_game(const struct retro_game_info *game); + +// Loads a "special" kind of game. Should not be used except in extreme cases. +bool retro_load_game_special( + unsigned game_type, + const struct retro_game_info *info, size_t num_info +); + +// Unloads a currently loaded game. +void retro_unload_game(void); + +// Gets region of game. +unsigned retro_get_region(void); + +// Gets region of memory. +void *retro_get_memory_data(unsigned id); +size_t retro_get_memory_size(unsigned id); + +#ifdef __cplusplus +} +#endif + +#endif diff --git a/mednafen/libretro/link.T b/mednafen/libretro/link.T new file mode 100644 index 0000000000..b0c262db9e --- /dev/null +++ b/mednafen/libretro/link.T @@ -0,0 +1,5 @@ +{ + global: retro_*; + local: *; +}; + diff --git a/mednafen/libretro/minifen/minifen.cpp b/mednafen/libretro/minifen/minifen.cpp new file mode 100644 index 0000000000..466ac53f83 --- /dev/null +++ b/mednafen/libretro/minifen/minifen.cpp @@ -0,0 +1,963 @@ +#include "mednafen.h" + +#include +#include +#include +#include +#include +#include + +#include "general.h" + +#include "state.h" +#include "video.h" +#ifdef NEED_DEINTERLACER +#include "video/Deinterlacer.h" +#endif + +#include "file.h" +#include "FileWrapper.h" + +#ifdef NEED_CD +#include "cdrom/cdromif.h" +#include "cdrom/CDUtility.h" +#endif + +#include "mempatcher.h" +#include "md5.h" +#include "clamp.h" +#ifdef NEED_RESAMPLER +#include "Fir_Resampler.h" +#endif + +MDFNGI *MDFNGameInfo = NULL; + +static double LastSoundMultiplier; + +static MDFN_PixelFormat last_pixel_format; +static double last_sound_rate; + +#ifdef NEED_DEINTERLACER +static bool PrevInterlaced; +static Deinterlacer deint; +#endif + +#ifdef NEED_CD +static std::vector CDInterfaces; // FIXME: Cleanup on error out. +#endif + +void MDFNI_CloseGame(void) +{ + if(MDFNGameInfo) + { + MDFN_FlushGameCheats(0); + + MDFNGameInfo->CloseGame(); + if(MDFNGameInfo->name) + { + free(MDFNGameInfo->name); + MDFNGameInfo->name=0; + } + MDFNMP_Kill(); + + MDFNGameInfo = NULL; + +#ifdef NEED_CD + for(unsigned i = 0; i < CDInterfaces.size(); i++) + delete CDInterfaces[i]; + CDInterfaces.clear(); +#endif + } + +#ifdef WANT_DEBUGGER + MDFNDBG_Kill(); +#endif +} + +std::vector MDFNSystems; +static std::list MDFNSystemsPrio; + +bool MDFNSystemsPrio_CompareFunc(MDFNGI *first, MDFNGI *second) +{ + if(first->ModulePriority > second->ModulePriority) + return(true); + + return(false); +} + +static void AddSystem(MDFNGI *system) +{ + MDFNSystems.push_back(system); +} + + +#ifdef NEED_CD +bool CDIF_DumpCD(const char *fn); + +void MDFNI_DumpModulesDef(const char *fn) +{ + FILE *fp = fopen(fn, "wb"); + + for(unsigned int i = 0; i < MDFNSystems.size(); i++) + { + fprintf(fp, "%s\n", MDFNSystems[i]->shortname); + fprintf(fp, "%s\n", MDFNSystems[i]->fullname); + fprintf(fp, "%d\n", MDFNSystems[i]->nominal_width); + fprintf(fp, "%d\n", MDFNSystems[i]->nominal_height); + } + + fclose(fp); +} + +static void ReadM3U(std::vector &file_list, std::string path, unsigned depth = 0) +{ + std::vector ret; + FileWrapper m3u_file(path.c_str(), FileWrapper::MODE_READ, _("M3U CD Set")); + std::string dir_path; + char linebuf[2048]; + + MDFN_GetFilePathComponents(path, &dir_path); + + while(m3u_file.get_line(linebuf, sizeof(linebuf))) + { + std::string efp; + + if(linebuf[0] == '#') continue; + MDFN_rtrim(linebuf); + if(linebuf[0] == 0) continue; + + efp = MDFN_EvalFIP(dir_path, std::string(linebuf)); + + if(efp.size() >= 4 && efp.substr(efp.size() - 4) == ".m3u") + { + if(efp == path) + throw(MDFN_Error(0, _("M3U at \"%s\" references self."), efp.c_str())); + + if(depth == 99) + throw(MDFN_Error(0, _("M3U load recursion too deep!"))); + + ReadM3U(file_list, efp, depth++); + } + else + file_list.push_back(efp); + } +} + +// TODO: LoadCommon() + +MDFNGI *MDFNI_LoadCD(const char *force_module, const char *devicename) +{ + uint8 LayoutMD5[16]; + + MDFNI_CloseGame(); + + LastSoundMultiplier = 1; + + MDFN_printf(_("Loading %s...\n\n"), devicename ? devicename : _("PHYSICAL CD")); + + try + { + if(devicename && strlen(devicename) > 4 && !strcasecmp(devicename + strlen(devicename) - 4, ".m3u")) + { + std::vector file_list; + + ReadM3U(file_list, devicename); + + for(unsigned i = 0; i < file_list.size(); i++) + CDInterfaces.push_back(new CDIF_MT(file_list[i].c_str())); + } + else + CDInterfaces.push_back(new CDIF_MT(devicename)); + } + catch(std::exception &e) + { + MDFND_PrintError(e.what()); + MDFN_PrintError(_("Error opening CD.")); + return(0); + } + + /* Print out a track list for all discs. */ + MDFN_indent(1); + for(unsigned i = 0; i < CDInterfaces.size(); i++) + { + CDUtility::TOC toc; + + CDInterfaces[i]->ReadTOC(&toc); + + MDFN_printf(_("CD %d Layout:\n"), i + 1); + MDFN_indent(1); + + for(int32 track = toc.first_track; track <= toc.last_track; track++) + { + MDFN_printf(_("Track %2d, LBA: %6d %s\n"), track, toc.tracks[track].lba, (toc.tracks[track].control & 0x4) ? "DATA" : "AUDIO"); + } + + MDFN_printf("Leadout: %6d\n", toc.tracks[100].lba); + MDFN_indent(-1); + MDFN_printf("\n"); + } + MDFN_indent(-1); + + // Calculate layout MD5. The system emulation LoadCD() code is free to ignore this value and calculate + // its own, or to use it to look up a game in its database. + { + md5_context layout_md5; + + layout_md5.starts(); + + for(unsigned i = 0; i < CDInterfaces.size(); i++) + { + CD_TOC toc; + + CDInterfaces[i]->ReadTOC(&toc); + + layout_md5.update_u32_as_lsb(toc.first_track); + layout_md5.update_u32_as_lsb(toc.last_track); + layout_md5.update_u32_as_lsb(toc.tracks[100].lba); + + for(uint32 track = toc.first_track; track <= toc.last_track; track++) + { + layout_md5.update_u32_as_lsb(toc.tracks[track].lba); + layout_md5.update_u32_as_lsb(toc.tracks[track].control & 0x4); + } + } + + layout_md5.finish(LayoutMD5); + } + + MDFNGameInfo = NULL; + + for(std::list::iterator it = MDFNSystemsPrio.begin(); it != MDFNSystemsPrio.end(); it++) + { + char tmpstr[256]; + trio_snprintf(tmpstr, 256, "%s.enable", (*it)->shortname); + + if(force_module) + { + if(!strcmp(force_module, (*it)->shortname)) + { + MDFNGameInfo = *it; + break; + } + } + else + { + // Is module enabled? + if(!MDFN_GetSettingB(tmpstr)) + continue; + + if(!(*it)->LoadCD || !(*it)->TestMagicCD) + continue; + + if((*it)->TestMagicCD(&CDInterfaces)) + { + MDFNGameInfo = *it; + break; + } + } + } + + if(!MDFNGameInfo) + { + if(force_module) + { + MDFN_PrintError(_("Unrecognized system \"%s\"!"), force_module); + return(0); + } + + // This code path should never be taken, thanks to "cdplay" + MDFN_PrintError(_("Could not find a system that supports this CD.")); + return(0); + } + + // This if statement will be true if force_module references a system without CDROM support. + if(!MDFNGameInfo->LoadCD) + { + MDFN_PrintError(_("Specified system \"%s\" doesn't support CDs!"), force_module); + return(0); + } + + MDFN_printf(_("Using module: %s(%s)\n\n"), MDFNGameInfo->shortname, MDFNGameInfo->fullname); + + + // TODO: include module name in hash + memcpy(MDFNGameInfo->MD5, LayoutMD5, 16); + + if(!(MDFNGameInfo->LoadCD(&CDInterfaces))) + { + for(unsigned i = 0; i < CDInterfaces.size(); i++) + delete CDInterfaces[i]; + CDInterfaces.clear(); + + MDFNGameInfo = NULL; + return(0); + } + + MDFNI_SetLayerEnableMask(~0ULL); + +#ifdef WANT_DEBUGGER + MDFNDBG_PostGameLoad(); +#endif + + MDFN_ResetMessages(); // Save state, status messages, etc. + + MDFN_LoadGameCheats(NULL); + MDFNMP_InstallReadPatches(); + + last_sound_rate = -1; + memset(&last_pixel_format, 0, sizeof(MDFN_PixelFormat)); + + return(MDFNGameInfo); +} +#endif + +// Return FALSE on fatal error(IPS file found but couldn't be applied), +// or TRUE on success(IPS patching succeeded, or IPS file not found). +static bool LoadIPS(MDFNFILE &GameFile, const char *path) +{ + return true; +} + +MDFNGI *MDFNI_LoadGame(const char *force_module, const char *name) +{ + MDFNFILE GameFile; + std::vector valid_iae; + +#ifdef NEED_CD + if(strlen(name) > 4 && (!strcasecmp(name + strlen(name) - 4, ".cue") || !strcasecmp(name + strlen(name) - 4, ".toc") || !strcasecmp(name + strlen(name) - 4, ".m3u"))) + return(MDFNI_LoadCD(force_module, name)); +#endif + + MDFNI_CloseGame(); + + LastSoundMultiplier = 1; + + MDFNGameInfo = NULL; + + MDFN_printf(_("Loading %s...\n"),name); + + MDFN_indent(1); + + // Construct a NULL-delimited list of known file extensions for MDFN_fopen() + for(unsigned int i = 0; i < MDFNSystems.size(); i++) + { + const FileExtensionSpecStruct *curexts = MDFNSystems[i]->FileExtensions; + + // If we're forcing a module, only look for extensions corresponding to that module + if(force_module && strcmp(MDFNSystems[i]->shortname, force_module)) + continue; + + if(curexts) + while(curexts->extension && curexts->description) + { + valid_iae.push_back(*curexts); + curexts++; + } + } + { + FileExtensionSpecStruct tmpext = { NULL, NULL }; + valid_iae.push_back(tmpext); + } + + if(!GameFile.Open(name, &valid_iae[0], _("game"))) + { + MDFNGameInfo = NULL; + return 0; + } + + if(!LoadIPS(GameFile, MDFN_MakeFName(MDFNMKF_IPS, 0, 0).c_str())) + { + MDFNGameInfo = NULL; + GameFile.Close(); + return(0); + } + + MDFNGameInfo = NULL; + + for(std::list::iterator it = MDFNSystemsPrio.begin(); it != MDFNSystemsPrio.end(); it++) //_unsigned int x = 0; x < MDFNSystems.size(); x++) + { + char tmpstr[256]; + trio_snprintf(tmpstr, 256, "%s.enable", (*it)->shortname); + + if(force_module) + { + if(!strcmp(force_module, (*it)->shortname)) + { + if(!(*it)->Load) + { + GameFile.Close(); +#ifdef NEED_CD + if((*it)->LoadCD) + MDFN_PrintError(_("Specified system only supports CD(physical, or image files, such as *.cue and *.toc) loading.")); + else +#endif + MDFN_PrintError(_("Specified system does not support normal file loading.")); + MDFN_indent(-1); + MDFNGameInfo = NULL; + return 0; + } + MDFNGameInfo = *it; + break; + } + } + else + { + // Is module enabled? + if(!MDFN_GetSettingB(tmpstr)) + continue; + + if(!(*it)->Load || !(*it)->TestMagic) + continue; + + if((*it)->TestMagic(name, &GameFile)) + { + MDFNGameInfo = *it; + break; + } + } + } + + if(!MDFNGameInfo) + { + GameFile.Close(); + + if(force_module) + MDFN_PrintError(_("Unrecognized system \"%s\"!"), force_module); + else + MDFN_PrintError(_("Unrecognized file format. Sorry.")); + + MDFN_indent(-1); + MDFNGameInfo = NULL; + return 0; + } + + MDFN_printf(_("Using module: %s(%s)\n\n"), MDFNGameInfo->shortname, MDFNGameInfo->fullname); + MDFN_indent(1); + + MDFNGameInfo->soundrate = 0; + MDFNGameInfo->name = NULL; + MDFNGameInfo->rotated = 0; + + // + // Load per-game settings + // + // Maybe we should make a "pgcfg" subdir, and automatically load all files in it? + // End load per-game settings + // + + if(MDFNGameInfo->Load(name, &GameFile) <= 0) + { + GameFile.Close(); + MDFN_indent(-2); + MDFNGameInfo = NULL; + return(0); + } + + MDFN_LoadGameCheats(NULL); + MDFNMP_InstallReadPatches(); + + MDFNI_SetLayerEnableMask(~0ULL); + +#ifdef WANT_DEBUGGER + MDFNDBG_PostGameLoad(); +#endif + + MDFN_ResetMessages(); // Save state, status messages, etc. + + MDFN_indent(-2); + + if(!MDFNGameInfo->name) + { + unsigned int x; + char *tmp; + + MDFNGameInfo->name = (UTF8 *)strdup(GetFNComponent(name)); + + for(x=0;xname);x++) + { + if(MDFNGameInfo->name[x] == '_') + MDFNGameInfo->name[x] = ' '; + } + if((tmp = strrchr((char *)MDFNGameInfo->name, '.'))) + *tmp = 0; + } + +#ifdef NEED_DEINTERLACER + PrevInterlaced = false; + deint.ClearState(); +#endif + + last_sound_rate = -1; + memset(&last_pixel_format, 0, sizeof(MDFN_PixelFormat)); + + return(MDFNGameInfo); +} + +#if defined(WANT_PSX_EMU) +extern MDFNGI EmulatedPSX; +#elif defined(WANT_PCE_EMU) +extern MDFNGI EmulatedPCE; +#elif defined(WANT_PCE_FAST_EMU) +extern MDFNGI EmulatedPCE_Fast; +#elif defined(WANT_WSWAN_EMU) +extern MDFNGI EmulatedWSwan; +#endif + +bool MDFNI_InitializeModules(const std::vector &ExternalSystems) +{ + static MDFNGI *InternalSystems[] = + { +#ifdef WANT_NES_EMU + &EmulatedNES, +#endif + +#ifdef WANT_SNES_EMU + &EmulatedSNES, +#endif + +#ifdef WANT_GB_EMU + &EmulatedGB, +#endif + +#ifdef WANT_GBA_EMU + &EmulatedGBA, +#endif + +#ifdef WANT_PCE_EMU + &EmulatedPCE, +#endif + +#ifdef WANT_PCE_FAST_EMU + &EmulatedPCE_Fast, +#endif + +#ifdef WANT_LYNX_EMU + &EmulatedLynx, +#endif + +#ifdef WANT_MD_EMU + &EmulatedMD, +#endif + +#ifdef WANT_PCFX_EMU + &EmulatedPCFX, +#endif + +#ifdef WANT_NGP_EMU + &EmulatedNGP, +#endif + +#ifdef WANT_PSX_EMU + &EmulatedPSX, +#endif + +#ifdef WANT_VB_EMU + &EmulatedVB, +#endif + +#ifdef WANT_WSWAN_EMU + &EmulatedWSwan, +#endif + +#ifdef WANT_SMS_EMU + &EmulatedSMS, + &EmulatedGG, +#endif + }; + std::string i_modules_string, e_modules_string; + + for(unsigned int i = 0; i < sizeof(InternalSystems) / sizeof(MDFNGI *); i++) + { + AddSystem(InternalSystems[i]); + if(i) + i_modules_string += " "; + i_modules_string += std::string(InternalSystems[i]->shortname); + } + + MDFNI_printf(_("Internal emulation modules: %s\n"), i_modules_string.c_str()); + + for(unsigned int i = 0; i < MDFNSystems.size(); i++) + MDFNSystemsPrio.push_back(MDFNSystems[i]); + + MDFNSystemsPrio.sort(MDFNSystemsPrio_CompareFunc); + +#ifdef NEED_CD + CDUtility::CDUtility_Init(); +#endif + + return(1); +} + +static std::string settings_file_path; +int MDFNI_Initialize(const char *basedir) +{ +#ifdef WANT_DEBUGGER + MDFNDBG_Init(); +#endif + + return(1); +} + +void MDFNI_Kill(void) +{ + /* save settings */ +} + +#if defined(NEED_RESAMPLER) +static double multiplier_save, volume_save; +static Fir_Resampler<16> ff_resampler; + +static void ProcessAudio(EmulateSpecStruct *espec) +{ + if(espec->SoundVolume != 1) + volume_save = espec->SoundVolume; + + if(espec->soundmultiplier != 1) + multiplier_save = espec->soundmultiplier; + + if(espec->SoundBuf && espec->SoundBufSize) + { + int16 *const SoundBuf = espec->SoundBuf + espec->SoundBufSizeALMS * MDFNGameInfo->soundchan; + int32 SoundBufSize = espec->SoundBufSize - espec->SoundBufSizeALMS; + const int32 SoundBufMaxSize = espec->SoundBufMaxSize - espec->SoundBufSizeALMS; + + if(multiplier_save != LastSoundMultiplier) + { + ff_resampler.time_ratio(multiplier_save, 0.9965); + LastSoundMultiplier = multiplier_save; + } + + if(multiplier_save != 1) + { + { + if(MDFNGameInfo->soundchan == 2) + { + for(int i = 0; i < SoundBufSize * 2; i++) + ff_resampler.buffer()[i] = SoundBuf[i]; + } + else + { + for(int i = 0; i < SoundBufSize; i++) + { + ff_resampler.buffer()[i * 2] = SoundBuf[i]; + ff_resampler.buffer()[i * 2 + 1] = 0; + } + } + ff_resampler.write(SoundBufSize * 2); + + int avail = ff_resampler.avail(); + int real_read = std::min((int)(SoundBufMaxSize * MDFNGameInfo->soundchan), avail); + + if(MDFNGameInfo->soundchan == 2) + SoundBufSize = ff_resampler.read(SoundBuf, real_read ) >> 1; + else + SoundBufSize = ff_resampler.read_mono_hack(SoundBuf, real_read ); + + avail -= real_read; + + if(avail > 0) + { + printf("ff_resampler.avail() > espec->SoundBufMaxSize * MDFNGameInfo->soundchan - %d\n", avail); + ff_resampler.clear(); + } + } + } + + if(volume_save != 1) + { + if(volume_save < 1) + { + int volume = (int)(16384 * volume_save); + + for(int i = 0; i < SoundBufSize * MDFNGameInfo->soundchan; i++) + SoundBuf[i] = (SoundBuf[i] * volume) >> 14; + } + else + { + int volume = (int)(256 * volume_save); + + for(int i = 0; i < SoundBufSize * MDFNGameInfo->soundchan; i++) + { + int temp = ((SoundBuf[i] * volume) >> 8) + 32768; + + temp = clamp_to_u16(temp); + + SoundBuf[i] = temp - 32768; + } + } + } + + // TODO: Optimize this. + if(MDFNGameInfo->soundchan == 2 && MDFN_GetSettingB(std::string(std::string(MDFNGameInfo->shortname) + ".forcemono").c_str())) + { + for(int i = 0; i < SoundBufSize * MDFNGameInfo->soundchan; i += 2) + { + // We should use division instead of arithmetic right shift for correctness(rounding towards 0 instead of negative infinitininintinity), but I like speed. + int32 mixed = (SoundBuf[i + 0] + SoundBuf[i + 1]) >> 1; + + SoundBuf[i + 0] = + SoundBuf[i + 1] = mixed; + } + } + + espec->SoundBufSize = espec->SoundBufSizeALMS + SoundBufSize; + } // end to: if(espec->SoundBuf && espec->SoundBufSize) +} +#else +static void ProcessAudio(EmulateSpecStruct *espec) +{ + if(espec->SoundBuf && espec->SoundBufSize) + { + int16 *const SoundBuf = espec->SoundBuf + espec->SoundBufSizeALMS * MDFNGameInfo->soundchan; + int32 SoundBufSize = espec->SoundBufSize - espec->SoundBufSizeALMS; + const int32 SoundBufMaxSize = espec->SoundBufMaxSize - espec->SoundBufSizeALMS; + + espec->SoundBufSize = espec->SoundBufSizeALMS + SoundBufSize; + } // end to: if(espec->SoundBuf && espec->SoundBufSize) +} +#endif + +void MDFN_MidSync(EmulateSpecStruct *espec) +{ + ProcessAudio(espec); + + MDFND_MidSync(espec); + + espec->SoundBufSizeALMS = espec->SoundBufSize; + espec->MasterCyclesALMS = espec->MasterCycles; +} + +void MDFNI_Emulate(EmulateSpecStruct *espec) +{ + // Initialize some espec member data to zero, to catch some types of bugs. + espec->DisplayRect.x = 0; + espec->DisplayRect.w = 0; + espec->DisplayRect.y = 0; + espec->DisplayRect.h = 0; + + espec->SoundBufSize = 0; + + espec->VideoFormatChanged = false; + espec->SoundFormatChanged = false; + + if(memcmp(&last_pixel_format, &espec->surface->format, sizeof(MDFN_PixelFormat))) + { + espec->VideoFormatChanged = TRUE; + + last_pixel_format = espec->surface->format; + } + + if(espec->SoundRate != last_sound_rate) + { + espec->SoundFormatChanged = true; + last_sound_rate = espec->SoundRate; + +#ifdef NEED_RESAMPLER + ff_resampler.buffer_size((espec->SoundRate / 2) * 2); +#endif + } + + espec->NeedSoundReverse = false; + + MDFNGameInfo->Emulate(espec); + +#ifdef NEED_DEINTERLACER + if(espec->InterlaceOn) + { + if(!PrevInterlaced) + deint.ClearState(); + + deint.Process(espec->surface, espec->DisplayRect, espec->LineWidths, espec->InterlaceField); + + PrevInterlaced = true; + + espec->InterlaceOn = false; + espec->InterlaceField = 0; + } + else + PrevInterlaced = false; +#endif + + ProcessAudio(espec); +} + +// This function should only be called for state rewinding. +// FIXME: Add a macro for SFORMAT structure access instead of direct access +int MDFN_RawInputStateAction(StateMem *sm, int load, int data_only) +{ + static const char *stringies[16] = { "RI00", "RI01", "RI02", "RI03", "RI04", "RI05", "RI06", "RI07", "RI08", "RI09", "RI0a", "RI0b", "RI0c", "RI0d", "RI0e", "RI0f" }; + SFORMAT StateRegs[17]; + int x; + + for(x = 0; x < 16; x++) + { + StateRegs[x].name = stringies[x]; + StateRegs[x].flags = 0; + + StateRegs[x].v = NULL; + StateRegs[x].size = 0; + } + + StateRegs[x].v = NULL; + StateRegs[x].size = 0; + StateRegs[x].name = NULL; + + int ret = MDFNSS_StateAction(sm, load, data_only, StateRegs, "rinp"); + + return(ret); +} + +static int curindent = 0; + +void MDFN_indent(int indent) +{ + curindent += indent; +} + +static uint8 lastchar = 0; +void MDFN_printf(const char *format, ...) throw() +{ + char *format_temp; + char *temp; + unsigned int x, newlen; + + va_list ap; + va_start(ap,format); + + + // First, determine how large our format_temp buffer needs to be. + uint8 lastchar_backup = lastchar; // Save lastchar! + for(newlen=x=0;xDoSimpleCommand(cmd); +} + +void MDFN_QSimpleCommand(int cmd) +{ +} + +void MDFNI_Power(void) +{ + MDFN_QSimpleCommand(MDFN_MSC_POWER); +} + +void MDFNI_Reset(void) +{ + MDFN_QSimpleCommand(MDFN_MSC_RESET); +} + +// Arcade-support functions +void MDFNI_ToggleDIPView(void) +{ + +} + +void MDFNI_ToggleDIP(int which) +{ + MDFN_QSimpleCommand(MDFN_MSC_TOGGLE_DIP0 + which); +} + +void MDFNI_InsertCoin(void) +{ + MDFN_QSimpleCommand(MDFN_MSC_INSERT_COIN); +} + +// Disk/Disc-based system support functions +void MDFNI_DiskInsert(int which) +{ + MDFN_QSimpleCommand(MDFN_MSC_INSERT_DISK0 + which); +} + +void MDFNI_DiskSelect() +{ + MDFN_QSimpleCommand(MDFN_MSC_SELECT_DISK); +} + +void MDFNI_DiskInsert() +{ + MDFN_QSimpleCommand(MDFN_MSC_INSERT_DISK); +} + +void MDFNI_DiskEject() +{ + MDFN_QSimpleCommand(MDFN_MSC_EJECT_DISK); +} + +void MDFNI_SetLayerEnableMask(uint64 mask) +{ + MDFNGameInfo->SetLayerEnableMask(mask); +} + +void MDFNI_SetInput(int port, const char *type, void *ptr, uint32 ptr_len_thingy) +{ + MDFNGameInfo->SetInput(port, type, ptr); +} diff --git a/mednafen/libretro/minifen/minifen_state.cpp b/mednafen/libretro/minifen/minifen_state.cpp new file mode 100644 index 0000000000..74fb94024f --- /dev/null +++ b/mednafen/libretro/minifen/minifen_state.cpp @@ -0,0 +1,542 @@ +/* Mednafen - Multi-system Emulator + * + * This program is free software; you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation; either version 2 of the License, or + * (at your option) any later version. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License + * along with this program; if not, write to the Free Software + * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA + */ + +#include "mednafen.h" +#include "mednafen-endian.h" + +#include + +#include +#include "driver.h" +#include "general.h" +#include "state.h" +#include "video.h" + +#define RLSB MDFNSTATE_RLSB //0x80000000 + +int32 smem_read(StateMem *st, void *buffer, uint32 len) +{ + if((len + st->loc) > st->len) + return(0); + + memcpy(buffer, st->data + st->loc, len); + st->loc += len; + + return(len); +} + +int32 smem_write(StateMem *st, void *buffer, uint32 len) +{ + if((len + st->loc) > st->malloced) + { + uint32 newsize = (st->malloced >= 32768) ? st->malloced : (st->initial_malloc ? st->initial_malloc : 32768); + + while(newsize < (len + st->loc)) + newsize *= 2; + st->data = (uint8 *)realloc(st->data, newsize); + st->malloced = newsize; + } + memcpy(st->data + st->loc, buffer, len); + st->loc += len; + + if(st->loc > st->len) st->len = st->loc; + + return(len); +} + +int32 smem_putc(StateMem *st, int value) +{ + uint8 tmpval = value; + if(smem_write(st, &tmpval, 1) != 1) + return(-1); + return(1); +} + +int32 smem_tell(StateMem *st) +{ + return(st->loc); +} + +int32 smem_seek(StateMem *st, uint32 offset, int whence) +{ + switch(whence) + { + case SEEK_SET: st->loc = offset; break; + case SEEK_END: st->loc = st->len - offset; break; + case SEEK_CUR: st->loc += offset; break; + } + + if(st->loc > st->len) + { + st->loc = st->len; + return(-1); + } + + if(st->loc < 0) + { + st->loc = 0; + return(-1); + } + + return(0); +} + +int smem_write32le(StateMem *st, uint32 b) +{ + uint8 s[4]; + s[0]=b; + s[1]=b>>8; + s[2]=b>>16; + s[3]=b>>24; + return((smem_write(st, s, 4)<4)?0:4); +} + +int smem_read32le(StateMem *st, uint32 *b) +{ + uint8 s[4]; + + if(smem_read(st, s, 4) < 4) + return(0); + + *b = s[0] | (s[1] << 8) | (s[2] << 16) | (s[3] << 24); + + return(4); +} + + +static bool ValidateSFStructure(SFORMAT *sf) +{ + SFORMAT *saved_sf = sf; + + while(sf->size || sf->name) + { + SFORMAT *sub_sf = saved_sf; + while(sub_sf->size || sub_sf->name) + { + if(sf != sub_sf) + { + if(!strncmp(sf->name, sub_sf->name, 32)) + fprintf(stderr, "Duplicate state variable name: %.32s\n", sf->name); + } + sub_sf++; + } + + sf++; + } + return(1); +} + + +static bool SubWrite(StateMem *st, SFORMAT *sf, const char *name_prefix = NULL) +{ + // FIXME? It's kind of slow, and we definitely don't want it on with state rewinding... + ValidateSFStructure(sf); + + while(sf->size || sf->name) // Size can sometimes be zero, so also check for the text name. These two should both be zero only at the end of a struct. + { + if(!sf->size || !sf->v) + { + sf++; + continue; + } + + if(sf->size == (uint32)~0) /* Link to another struct. */ + { + if(!SubWrite(st, (SFORMAT *)sf->v, name_prefix)) + return(0); + + sf++; + continue; + } + + int32 bytesize = sf->size; + + char nameo[1 + 256]; + int slen; + + slen = trio_snprintf(nameo + 1, 256, "%s%s", name_prefix ? name_prefix : "", sf->name); + nameo[0] = slen; + + if(slen >= 255) + { + fprintf(stderr, "Warning: state variable name possibly too long: %s %s %s %d\n", sf->name, name_prefix, nameo, slen); + slen = 255; + } + + smem_write(st, nameo, 1 + nameo[0]); + smem_write32le(st, bytesize); + + /* Flip the byte order... */ + if(sf->flags & MDFNSTATE_BOOL) + { + + } + else if(sf->flags & MDFNSTATE_RLSB64) + Endian_A64_NE_to_LE(sf->v, bytesize / sizeof(uint64)); + else if(sf->flags & MDFNSTATE_RLSB32) + Endian_A32_NE_to_LE(sf->v, bytesize / sizeof(uint32)); + else if(sf->flags & MDFNSTATE_RLSB16) + Endian_A16_NE_to_LE(sf->v, bytesize / sizeof(uint16)); + else if(sf->flags & RLSB) + Endian_V_NE_to_LE(sf->v, bytesize); + + // Special case for the evil bool type, to convert bool to 1-byte elements. + // Don't do it if we're only saving the raw data. + if(sf->flags & MDFNSTATE_BOOL) + { + for(int32 bool_monster = 0; bool_monster < bytesize; bool_monster++) + { + uint8 tmp_bool = ((bool *)sf->v)[bool_monster]; + //printf("Bool write: %.31s\n", sf->name); + smem_write(st, &tmp_bool, 1); + } + } + else + smem_write(st, (uint8 *)sf->v, bytesize); + + /* Now restore the original byte order. */ + if(sf->flags & MDFNSTATE_BOOL) + { + + } + else if(sf->flags & MDFNSTATE_RLSB64) + Endian_A64_LE_to_NE(sf->v, bytesize / sizeof(uint64)); + else if(sf->flags & MDFNSTATE_RLSB32) + Endian_A32_LE_to_NE(sf->v, bytesize / sizeof(uint32)); + else if(sf->flags & MDFNSTATE_RLSB16) + Endian_A16_LE_to_NE(sf->v, bytesize / sizeof(uint16)); + else if(sf->flags & RLSB) + Endian_V_LE_to_NE(sf->v, bytesize); + sf++; + } + + return(TRUE); +} + +static int WriteStateChunk(StateMem *st, const char *sname, SFORMAT *sf) +{ + int32 data_start_pos; + int32 end_pos; + + uint8 sname_tmp[32]; + + memset(sname_tmp, 0, sizeof(sname_tmp)); + strncpy((char *)sname_tmp, sname, 32); + + if(strlen(sname) > 32) + fprintf(stderr, "Warning: section name is too long: %s\n", sname); + + smem_write(st, sname_tmp, 32); + + smem_write32le(st, 0); // We'll come back and write this later. + + data_start_pos = smem_tell(st); + + if(!SubWrite(st, sf)) + return(0); + + end_pos = smem_tell(st); + + smem_seek(st, data_start_pos - 4, SEEK_SET); + smem_write32le(st, end_pos - data_start_pos); + smem_seek(st, end_pos, SEEK_SET); + + return(end_pos - data_start_pos); +} + +struct compare_cstr +{ + bool operator()(const char *s1, const char *s2) const + { + return(strcmp(s1, s2) < 0); + } +}; + +typedef std::map SFMap_t; + +static void MakeSFMap(SFORMAT *sf, SFMap_t &sfmap) +{ + while(sf->size || sf->name) // Size can sometimes be zero, so also check for the text name. These two should both be zero only at the end of a struct. + { + if(!sf->size || !sf->v) + { + sf++; + continue; + } + + if(sf->size == (uint32)~0) /* Link to another SFORMAT structure. */ + MakeSFMap((SFORMAT *)sf->v, sfmap); + else + { + assert(sf->name); + + if(sfmap.find(sf->name) != sfmap.end()) + fprintf(stderr, "Duplicate save state variable in internal emulator structures(CLUB THE PROGRAMMERS WITH BREADSTICKS): %s\n", sf->name); + + sfmap[sf->name] = sf; + } + + sf++; + } +} + +// Fast raw chunk reader +static void DOReadChunk(StateMem *st, SFORMAT *sf) +{ + while(sf->size || sf->name) // Size can sometimes be zero, so also check for the text name. + // These two should both be zero only at the end of a struct. + { + if(!sf->size || !sf->v) + { + sf++; + continue; + } + + if(sf->size == (uint32) ~0) // Link to another SFORMAT struct + { + DOReadChunk(st, (SFORMAT *)sf->v); + sf++; + continue; + } + + int32 bytesize = sf->size; + + // Loading raw data, bool types are stored as they appear in memory, not as single bytes in the full state format. + // In the SFORMAT structure, the size member for bool entries is the number of bool elements, not the total in-memory size, + // so we adjust it here. + if(sf->flags & MDFNSTATE_BOOL) + bytesize *= sizeof(bool); + + smem_read(st, (uint8 *)sf->v, bytesize); + sf++; + } +} + +static int ReadStateChunk(StateMem *st, SFORMAT *sf, int size) +{ + int temp; + + { + SFMap_t sfmap; + SFMap_t sfmap_found; // Used for identifying variables that are missing in the save state. + + MakeSFMap(sf, sfmap); + + temp = smem_tell(st); + while(smem_tell(st) < (temp + size)) + { + uint32 recorded_size; // In bytes + uint8 toa[1 + 256]; // Don't change to char unless cast toa[0] to unsigned to smem_read() and other places. + + if(smem_read(st, toa, 1) != 1) + { + puts("Unexpected EOF"); + return(0); + } + + if(smem_read(st, toa + 1, toa[0]) != toa[0]) + { + puts("Unexpected EOF?"); + return 0; + } + + toa[1 + toa[0]] = 0; + + smem_read32le(st, &recorded_size); + + SFMap_t::iterator sfmit; + + sfmit = sfmap.find((char *)toa + 1); + + if(sfmit != sfmap.end()) + { + SFORMAT *tmp = sfmit->second; + uint32 expected_size = tmp->size; // In bytes + + if(recorded_size != expected_size) + { + fprintf(stderr, "Variable in save state wrong size: %s. Need: %d, got: %d\n", toa + 1, expected_size, recorded_size); + if(smem_seek(st, recorded_size, SEEK_CUR) < 0) + { + puts("Seek error"); + return(0); + } + } + else + { + sfmap_found[tmp->name] = tmp; + + smem_read(st, (uint8 *)tmp->v, expected_size); + + if(tmp->flags & MDFNSTATE_BOOL) + { + // Converting downwards is necessary for the case of sizeof(bool) > 1 + for(int32 bool_monster = expected_size - 1; bool_monster >= 0; bool_monster--) + { + ((bool *)tmp->v)[bool_monster] = ((uint8 *)tmp->v)[bool_monster]; + } + } + if(tmp->flags & MDFNSTATE_RLSB64) + Endian_A64_LE_to_NE(tmp->v, expected_size / sizeof(uint64)); + else if(tmp->flags & MDFNSTATE_RLSB32) + Endian_A32_LE_to_NE(tmp->v, expected_size / sizeof(uint32)); + else if(tmp->flags & MDFNSTATE_RLSB16) + Endian_A16_LE_to_NE(tmp->v, expected_size / sizeof(uint16)); + else if(tmp->flags & RLSB) + Endian_V_LE_to_NE(tmp->v, expected_size); + } + } + else + { + fprintf(stderr, "Unknown variable in save state: %s\n", toa + 1); + if(smem_seek(st, recorded_size, SEEK_CUR) < 0) + { + puts("Seek error"); + return(0); + } + } + } // while(...) + + for(SFMap_t::const_iterator it = sfmap.begin(); it != sfmap.end(); it++) + { + if(sfmap_found.find(it->second->name) == sfmap_found.end()) + { + fprintf(stderr, "Variable missing from save state: %s\n", it->second->name); + } + } + + assert(smem_tell(st) == (temp + size)); + } + return 1; +} + +static int CurrentState = 0; + +/* This function is called by the game driver(NES, GB, GBA) to save a state. */ +int MDFNSS_StateAction(StateMem *st, int load, int data_only, std::vector §ions) +{ + std::vector::iterator section; + + if(load) + { + { + char sname[32]; + + for(section = sections.begin(); section != sections.end(); section++) + { + int found = 0; + uint32 tmp_size; + uint32 total = 0; + + while(smem_read(st, (uint8 *)sname, 32) == 32) + { + if(smem_read32le(st, &tmp_size) != 4) + return(0); + + total += tmp_size + 32 + 4; + + // Yay, we found the section + if(!strncmp(sname, section->name, 32)) + { + if(!ReadStateChunk(st, section->sf, tmp_size)) + { + fprintf(stderr, "Error reading chunk: %s\n", section->name); + return(0); + } + found = 1; + break; + } + else + { + if(smem_seek(st, tmp_size, SEEK_CUR) < 0) + { + puts("Chunk seek failure"); + return(0); + } + } + } + if(smem_seek(st, -total, SEEK_CUR) < 0) + { + puts("Reverse seek error"); + return(0); + } + if(!found && !section->optional) // Not found. We are sad! + { + fprintf(stderr, "Section missing: %.32s\n", section->name); + return(0); + } + } + } + } + else + { + for(section = sections.begin(); section != sections.end(); section++) + { + if(!WriteStateChunk(st, section->name, section->sf)) + return(0); + } + } + + return(1); +} + +int MDFNSS_StateAction(StateMem *st, int load, int data_only, SFORMAT *sf, const char *name, bool optional) +{ + std::vector love; + + love.push_back(SSDescriptor(sf, name, optional)); + return(MDFNSS_StateAction(st, load, 0, love)); +} + +int MDFNSS_SaveSM(StateMem *st, int, int, const MDFN_Surface*, const MDFN_Rect*, const MDFN_Rect*) +{ + static const char *header_magic = "MDFNSVST"; + uint8 header[32]; + int neowidth = 0, neoheight = 0; + + memset(header, 0, sizeof(header)); + memcpy(header, header_magic, 8); + + MDFN_en32lsb(header + 16, MEDNAFEN_VERSION_NUMERIC); + MDFN_en32lsb(header + 24, neowidth); + MDFN_en32lsb(header + 28, neoheight); + smem_write(st, header, 32); + + if(!MDFNGameInfo->StateAction(st, 0, 0)) + return(0); + + uint32 sizy = smem_tell(st); + smem_seek(st, 16 + 4, SEEK_SET); + smem_write32le(st, sizy); + + return(1); +} + +int MDFNSS_LoadSM(StateMem *st, int, int) +{ + uint8 header[32]; + uint32 stateversion; + + smem_read(st, header, 32); + + if(memcmp(header, "MEDNAFENSVESTATE", 16) && memcmp(header, "MDFNSVST", 8)) + return(0); + + stateversion = MDFN_de32lsb(header + 16); + + return(MDFNGameInfo->StateAction(st, stateversion, 0)); +} diff --git a/mednafen/libretro/scrc32.cpp b/mednafen/libretro/scrc32.cpp new file mode 100644 index 0000000000..5867aa5d98 --- /dev/null +++ b/mednafen/libretro/scrc32.cpp @@ -0,0 +1,74 @@ +static const unsigned long crc_table[256] = { + 0x00000000L, 0x77073096L, 0xee0e612cL, 0x990951baL, 0x076dc419L, + 0x706af48fL, 0xe963a535L, 0x9e6495a3L, 0x0edb8832L, 0x79dcb8a4L, + 0xe0d5e91eL, 0x97d2d988L, 0x09b64c2bL, 0x7eb17cbdL, 0xe7b82d07L, + 0x90bf1d91L, 0x1db71064L, 0x6ab020f2L, 0xf3b97148L, 0x84be41deL, + 0x1adad47dL, 0x6ddde4ebL, 0xf4d4b551L, 0x83d385c7L, 0x136c9856L, + 0x646ba8c0L, 0xfd62f97aL, 0x8a65c9ecL, 0x14015c4fL, 0x63066cd9L, + 0xfa0f3d63L, 0x8d080df5L, 0x3b6e20c8L, 0x4c69105eL, 0xd56041e4L, + 0xa2677172L, 0x3c03e4d1L, 0x4b04d447L, 0xd20d85fdL, 0xa50ab56bL, + 0x35b5a8faL, 0x42b2986cL, 0xdbbbc9d6L, 0xacbcf940L, 0x32d86ce3L, + 0x45df5c75L, 0xdcd60dcfL, 0xabd13d59L, 0x26d930acL, 0x51de003aL, + 0xc8d75180L, 0xbfd06116L, 0x21b4f4b5L, 0x56b3c423L, 0xcfba9599L, + 0xb8bda50fL, 0x2802b89eL, 0x5f058808L, 0xc60cd9b2L, 0xb10be924L, + 0x2f6f7c87L, 0x58684c11L, 0xc1611dabL, 0xb6662d3dL, 0x76dc4190L, + 0x01db7106L, 0x98d220bcL, 0xefd5102aL, 0x71b18589L, 0x06b6b51fL, + 0x9fbfe4a5L, 0xe8b8d433L, 0x7807c9a2L, 0x0f00f934L, 0x9609a88eL, + 0xe10e9818L, 0x7f6a0dbbL, 0x086d3d2dL, 0x91646c97L, 0xe6635c01L, + 0x6b6b51f4L, 0x1c6c6162L, 0x856530d8L, 0xf262004eL, 0x6c0695edL, + 0x1b01a57bL, 0x8208f4c1L, 0xf50fc457L, 0x65b0d9c6L, 0x12b7e950L, + 0x8bbeb8eaL, 0xfcb9887cL, 0x62dd1ddfL, 0x15da2d49L, 0x8cd37cf3L, + 0xfbd44c65L, 0x4db26158L, 0x3ab551ceL, 0xa3bc0074L, 0xd4bb30e2L, + 0x4adfa541L, 0x3dd895d7L, 0xa4d1c46dL, 0xd3d6f4fbL, 0x4369e96aL, + 0x346ed9fcL, 0xad678846L, 0xda60b8d0L, 0x44042d73L, 0x33031de5L, + 0xaa0a4c5fL, 0xdd0d7cc9L, 0x5005713cL, 0x270241aaL, 0xbe0b1010L, + 0xc90c2086L, 0x5768b525L, 0x206f85b3L, 0xb966d409L, 0xce61e49fL, + 0x5edef90eL, 0x29d9c998L, 0xb0d09822L, 0xc7d7a8b4L, 0x59b33d17L, + 0x2eb40d81L, 0xb7bd5c3bL, 0xc0ba6cadL, 0xedb88320L, 0x9abfb3b6L, + 0x03b6e20cL, 0x74b1d29aL, 0xead54739L, 0x9dd277afL, 0x04db2615L, + 0x73dc1683L, 0xe3630b12L, 0x94643b84L, 0x0d6d6a3eL, 0x7a6a5aa8L, + 0xe40ecf0bL, 0x9309ff9dL, 0x0a00ae27L, 0x7d079eb1L, 0xf00f9344L, + 0x8708a3d2L, 0x1e01f268L, 0x6906c2feL, 0xf762575dL, 0x806567cbL, + 0x196c3671L, 0x6e6b06e7L, 0xfed41b76L, 0x89d32be0L, 0x10da7a5aL, + 0x67dd4accL, 0xf9b9df6fL, 0x8ebeeff9L, 0x17b7be43L, 0x60b08ed5L, + 0xd6d6a3e8L, 0xa1d1937eL, 0x38d8c2c4L, 0x4fdff252L, 0xd1bb67f1L, + 0xa6bc5767L, 0x3fb506ddL, 0x48b2364bL, 0xd80d2bdaL, 0xaf0a1b4cL, + 0x36034af6L, 0x41047a60L, 0xdf60efc3L, 0xa867df55L, 0x316e8eefL, + 0x4669be79L, 0xcb61b38cL, 0xbc66831aL, 0x256fd2a0L, 0x5268e236L, + 0xcc0c7795L, 0xbb0b4703L, 0x220216b9L, 0x5505262fL, 0xc5ba3bbeL, + 0xb2bd0b28L, 0x2bb45a92L, 0x5cb36a04L, 0xc2d7ffa7L, 0xb5d0cf31L, + 0x2cd99e8bL, 0x5bdeae1dL, 0x9b64c2b0L, 0xec63f226L, 0x756aa39cL, + 0x026d930aL, 0x9c0906a9L, 0xeb0e363fL, 0x72076785L, 0x05005713L, + 0x95bf4a82L, 0xe2b87a14L, 0x7bb12baeL, 0x0cb61b38L, 0x92d28e9bL, + 0xe5d5be0dL, 0x7cdcefb7L, 0x0bdbdf21L, 0x86d3d2d4L, 0xf1d4e242L, + 0x68ddb3f8L, 0x1fda836eL, 0x81be16cdL, 0xf6b9265bL, 0x6fb077e1L, + 0x18b74777L, 0x88085ae6L, 0xff0f6a70L, 0x66063bcaL, 0x11010b5cL, + 0x8f659effL, 0xf862ae69L, 0x616bffd3L, 0x166ccf45L, 0xa00ae278L, + 0xd70dd2eeL, 0x4e048354L, 0x3903b3c2L, 0xa7672661L, 0xd06016f7L, + 0x4969474dL, 0x3e6e77dbL, 0xaed16a4aL, 0xd9d65adcL, 0x40df0b66L, + 0x37d83bf0L, 0xa9bcae53L, 0xdebb9ec5L, 0x47b2cf7fL, 0x30b5ffe9L, + 0xbdbdf21cL, 0xcabac28aL, 0x53b39330L, 0x24b4a3a6L, 0xbad03605L, + 0xcdd70693L, 0x54de5729L, 0x23d967bfL, 0xb3667a2eL, 0xc4614ab8L, + 0x5d681b02L, 0x2a6f2b94L, 0xb40bbe37L, 0xc30c8ea1L, 0x5a05df1bL, + 0x2d02ef8dL +}; + +#define DO1_CRC32(buf) crc = crc_table[((int)crc ^ (*buf++)) & 0xff] ^ (crc >> 8); +#define DO2_CRC32(buf) DO1_CRC32(buf); DO1_CRC32(buf); +#define DO4_CRC32(buf) DO2_CRC32(buf); DO2_CRC32(buf); +#define DO8_CRC32(buf) DO4_CRC32(buf); DO4_CRC32(buf); + +static unsigned long crc32(unsigned long crc, const unsigned char *buf, unsigned int len) +{ + if (buf == 0) return 0L; + crc = crc ^ 0xffffffffL; + while (len >= 8) + { + DO8_CRC32(buf); + len -= 8; + } + if (len) do { + DO1_CRC32(buf); + } while (--len); + return crc ^ 0xffffffffL; +} diff --git a/mednafen/libretro/scrc32.h b/mednafen/libretro/scrc32.h new file mode 100644 index 0000000000..ff6162097e --- /dev/null +++ b/mednafen/libretro/scrc32.h @@ -0,0 +1,6 @@ +#ifndef __LIBRETRO_CRC32_H +#define __LIBRETRO_CRC32_H + +unsigned long crc32(unsigned long crc, const unsigned char *buf, unsigned int len); + +#endif /* __LIBRETRO_CRC32_H diff --git a/mednafen/libretro/stubs.cpp b/mednafen/libretro/stubs.cpp new file mode 100644 index 0000000000..5ce7c31f0d --- /dev/null +++ b/mednafen/libretro/stubs.cpp @@ -0,0 +1,165 @@ +#include "mednafen-types.h" +#include "mednafen.h" +#include "md5.h" +#include "git.h" +#include "general.h" +#include "mednafen-driver.h" + +#ifndef _WIN32 +#include +#endif + +#include + +#ifdef _WIN32 +#include +#else +#include +#endif + +#ifdef NEED_THREADING +#include "thread.h" +#endif + +// Stubs + +void MDFND_Sleep(unsigned int time) +{ +#ifdef _WIN32 + Sleep(time); +#else + usleep(time * 1000); +#endif +} + +extern std::string retro_base_directory; +extern std::string retro_base_name; + +#ifdef _WIN32 +static void sanitize_path(std::string &path) +{ + size_t size = path.size(); + for (size_t i = 0; i < size; i++) + if (path[i] == '/') + path[i] = '\\'; +} +#endif + +// Use a simpler approach to make sure that things go right for libretro. +std::string MDFN_MakeFName(MakeFName_Type type, int id1, const char *cd1) +{ + std::string ret; + switch (type) + { + case MDFNMKF_SAV: + ret = retro_base_directory + + std::string(PSS) + + retro_base_name + + std::string(".") + + md5_context::asciistr(MDFNGameInfo->MD5, 0) + + std::string(".") + + std::string(cd1); + break; + case MDFNMKF_FIRMWARE: + ret = std::string(cd1); + break; + default: + break; + } + +#ifdef _WIN32 + sanitize_path(ret); // Because Windows path handling is mongoloid. +#endif + fprintf(stderr, "[Mednafen]: Path request: %s\n", ret.c_str()); + return ret; +} + +void MDFND_DispMessage(unsigned char *str) +{ + std::cerr << str; +} + +void MDFND_Message(const char *str) +{ + std::cerr << str; +} + +void MDFND_MidSync(const EmulateSpecStruct *) +{} + +void MDFND_PrintError(const char* err) +{ + std::cerr << err; +} + +uint32 MDFND_GetTime() +{ + static bool first = true; + static uint32_t start_ms; + +#ifdef _WIN32 + DWORD ms = timeGetTime(); + if (first) + { + start_ms = ms; + first = false; + } +#else + struct timeval val; + gettimeofday(&val, NULL); + uint32_t ms = val.tv_sec * 1000 + val.tv_usec / 1000; + + if (first) + { + start_ms = ms; + first = false; + } +#endif + + return ms - start_ms; +} + +#ifdef NEED_THREADING +MDFN_Thread *MDFND_CreateThread(int (*fn)(void *), void *data) +{ + return (MDFN_Thread*)sthread_create((void (*)(void*))fn, data); +} + +void MDFND_WaitThread(MDFN_Thread *thr, int *val) +{ + sthread_join((sthread_t*)thr); + + if (val) + { + *val = 0; + fprintf(stderr, "WaitThread relies on return value.\n"); + } +} + +void MDFND_KillThread(MDFN_Thread *) +{ + fprintf(stderr, "Killing a thread is a BAD IDEA!\n"); +} + +MDFN_Mutex *MDFND_CreateMutex() +{ + return (MDFN_Mutex*)slock_new(); +} + +void MDFND_DestroyMutex(MDFN_Mutex *lock) +{ + slock_free((slock_t*)lock); +} + +int MDFND_LockMutex(MDFN_Mutex *lock) +{ + slock_lock((slock_t*)lock); + return 0; +} + +int MDFND_UnlockMutex(MDFN_Mutex *lock) +{ + slock_unlock((slock_t*)lock); + return 0; +} +#endif diff --git a/mednafen/libretro/thread.cpp b/mednafen/libretro/thread.cpp new file mode 100644 index 0000000000..85c5a6522b --- /dev/null +++ b/mednafen/libretro/thread.cpp @@ -0,0 +1,295 @@ +/* RetroArch - A frontend for libretro. + * Copyright (C) 2010-2012 - Hans-Kristian Arntzen + * + * RetroArch is free software: you can redistribute it and/or modify it under the terms + * of the GNU General Public License as published by the Free Software Found- + * ation, either version 3 of the License, or (at your option) any later version. + * + * RetroArch is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; + * without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR + * PURPOSE. See the GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License along with RetroArch. + * If not, see . + */ + +#include "thread.h" +#include + +#if defined(_WIN32) && !defined(_XBOX) +#define WIN32_LEAN_AND_MEAN +#include +#elif defined(_XBOX) +#include +#elif defined(GEKKO) +#include "thread/gx_pthread.h" +#else +#include +#include +#endif + +#ifdef __MACH__ +#include +#include +#endif + +struct thread_data +{ + void (*func)(void*); + void *userdata; +}; + +#ifdef _WIN32 + +struct sthread +{ + HANDLE thread; +}; + +static DWORD CALLBACK thread_wrap(void *data_) +{ + struct thread_data *data = (struct thread_data*)data_; + data->func(data->userdata); + free(data); + return 0; +} + +sthread_t *sthread_create(void (*thread_func)(void*), void *userdata) +{ + sthread_t *thread = (sthread_t*)calloc(1, sizeof(*thread)); + if (!thread) + return NULL; + + struct thread_data *data = (struct thread_data*)calloc(1, sizeof(*data)); + if (!data) + { + free(thread); + return NULL; + } + + data->func = thread_func; + data->userdata = userdata; + + thread->thread = CreateThread(NULL, 0, thread_wrap, data, 0, NULL); + if (!thread->thread) + { + free(data); + free(thread); + return NULL; + } + + return thread; +} + +void sthread_join(sthread_t *thread) +{ + WaitForSingleObject(thread->thread, INFINITE); + CloseHandle(thread->thread); + free(thread); +} + +struct slock +{ + CRITICAL_SECTION lock; +}; + +slock_t *slock_new(void) +{ + slock_t *lock = (slock_t*)calloc(1, sizeof(*lock)); + if (!lock) + return NULL; + + InitializeCriticalSection(&lock->lock); + return lock; +} + +void slock_free(slock_t *lock) +{ + DeleteCriticalSection(&lock->lock); + free(lock); +} + +void slock_lock(slock_t *lock) +{ + EnterCriticalSection(&lock->lock); +} + +void slock_unlock(slock_t *lock) +{ + LeaveCriticalSection(&lock->lock); +} + +struct scond +{ + HANDLE event; +}; + +scond_t *scond_new(void) +{ + scond_t *cond = (scond_t*)calloc(1, sizeof(*cond)); + if (!cond) + return NULL; + + cond->event = CreateEvent(NULL, FALSE, FALSE, NULL); + if (!cond->event) + { + free(cond); + return NULL; + } + + return cond; +} + +void scond_wait(scond_t *cond, slock_t *lock) +{ + WaitForSingleObject(cond->event, 0); + slock_unlock(lock); + + WaitForSingleObject(cond->event, INFINITE); + + slock_lock(lock); +} + +int scond_wait_timeout(scond_t *cond, slock_t *lock, unsigned timeout_ms) +{ + WaitForSingleObject(cond->event, 0); + slock_unlock(lock); + + DWORD res = WaitForSingleObject(cond->event, timeout_ms); + + slock_lock(lock); + return res == WAIT_OBJECT_0; +} + +void scond_signal(scond_t *cond) +{ + SetEvent(cond->event); +} + +void scond_free(scond_t *cond) +{ + CloseHandle(cond->event); + free(cond); +} + +#else + +struct sthread +{ + pthread_t id; +}; + +static void *thread_wrap(void *data_) +{ + struct thread_data *data = (struct thread_data*)data_; + data->func(data->userdata); + free(data); + return NULL; +} + +sthread_t *sthread_create(void (*thread_func)(void*), void *userdata) +{ + sthread_t *thr = (sthread_t*)calloc(1, sizeof(*thr)); + if (!thr) + return NULL; + + struct thread_data *data = (struct thread_data*)calloc(1, sizeof(*data)); + if (!data) + { + free(thr); + return NULL; + } + + data->func = thread_func; + data->userdata = userdata; + + if (pthread_create(&thr->id, NULL, thread_wrap, data) < 0) + { + free(data); + free(thr); + return NULL; + } + + return thr; +} + +void sthread_join(sthread_t *thread) +{ + pthread_join(thread->id, NULL); + free(thread); +} + +struct slock +{ + pthread_mutex_t lock; +}; + +slock_t *slock_new(void) +{ + slock_t *lock = (slock_t*)calloc(1, sizeof(*lock)); + if (!lock) + return NULL; + + if (pthread_mutex_init(&lock->lock, NULL) < 0) + { + free(lock); + return NULL; + } + + return lock; +} + +void slock_free(slock_t *lock) +{ + pthread_mutex_destroy(&lock->lock); + free(lock); +} + +void slock_lock(slock_t *lock) +{ + pthread_mutex_lock(&lock->lock); +} + +void slock_unlock(slock_t *lock) +{ + pthread_mutex_unlock(&lock->lock); +} + +struct scond +{ + pthread_cond_t cond; +}; + +scond_t *scond_new(void) +{ + scond_t *cond = (scond_t*)calloc(1, sizeof(*cond)); + if (!cond) + return NULL; + + if (pthread_cond_init(&cond->cond, NULL) < 0) + { + free(cond); + return NULL; + } + + return cond; +} + +void scond_free(scond_t *cond) +{ + pthread_cond_destroy(&cond->cond); + free(cond); +} + +void scond_wait(scond_t *cond, slock_t *lock) +{ + pthread_cond_wait(&cond->cond, &lock->lock); +} + +void scond_signal(scond_t *cond) +{ + pthread_cond_signal(&cond->cond); +} + +#endif + diff --git a/mednafen/libretro/thread.h b/mednafen/libretro/thread.h new file mode 100644 index 0000000000..3919278b1e --- /dev/null +++ b/mednafen/libretro/thread.h @@ -0,0 +1,46 @@ +/* RetroArch - A frontend for libretro. + * Copyright (C) 2010-2012 - Hans-Kristian Arntzen + * Copyright (C) 2011-2012 - Daniel De Matteis + * + * RetroArch is free software: you can redistribute it and/or modify it under the terms + * of the GNU General Public License as published by the Free Software Found- + * ation, either version 3 of the License, or (at your option) any later version. + * + * RetroArch is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; + * without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR + * PURPOSE. See the GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License along with RetroArch. + * If not, see . + */ + +#ifndef __LIBRETRO_THREAD_H +#define __LIBRETRO_THREAD_H + +// Implements the bare minimum needed for RetroArch. :) + +typedef struct sthread sthread_t; + +// Threading +sthread_t *sthread_create(void (*thread_func)(void*), void *userdata); +void sthread_join(sthread_t *thread); + +// Mutexes +typedef struct slock slock_t; + +slock_t *slock_new(void); +void slock_free(slock_t *lock); + +void slock_lock(slock_t *lock); +void slock_unlock(slock_t *lock); + +// Condition variables. +typedef struct scond scond_t; + +scond_t *scond_new(void); +void scond_free(scond_t *cond); + +void scond_wait(scond_t *cond, slock_t *lock); +void scond_signal(scond_t *cond); + +#endif /* __LIBRETRO_THREAD_H */ diff --git a/mednafen/libretro/thread/gx_pthread.h b/mednafen/libretro/thread/gx_pthread.h new file mode 100644 index 0000000000..c014b6291e --- /dev/null +++ b/mednafen/libretro/thread/gx_pthread.h @@ -0,0 +1,96 @@ +/* RetroArch - A frontend for libretro. + * Copyright (C) 2010-2012 - Hans-Kristian Arntzen + * Copyright (C) 2011-2012 - Daniel De Matteis + * + * RetroArch is free software: you can redistribute it and/or modify it under the terms + * of the GNU General Public License as published by the Free Software Found- + * ation, either version 3 of the License, or (at your option) any later version. + * + * RetroArch is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; + * without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR + * PURPOSE. See the GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License along with RetroArch. + * If not, see . + */ + +#ifndef _GX_PTHREAD_WRAP_GX_ +#define _GX_PTHREAD_WRAP_GX_ + +#include +#include +#include + +#define STACKSIZE (8 * 1024) + +typedef lwp_t pthread_t; +typedef mutex_t pthread_mutex_t; +typedef void* pthread_mutexattr_t; +typedef int pthread_attr_t; +typedef cond_t pthread_cond_t; +typedef cond_t pthread_condattr_t; + +static inline int pthread_create(pthread_t *thread, const pthread_attr_t *attr, void *(*start_routine)(void*), void *arg) +{ + *thread = 0; + return LWP_CreateThread(thread, start_routine, arg, 0, STACKSIZE, 64); +} + +static inline int pthread_mutex_init(pthread_mutex_t *mutex, const pthread_mutexattr_t *attr) +{ + return LWP_MutexInit(mutex, 0); +} + +static inline int pthread_mutex_destroy(pthread_mutex_t *mutex) +{ + return LWP_MutexDestroy(*mutex); +} + +static inline int pthread_mutex_lock(pthread_mutex_t *mutex) +{ + return LWP_MutexLock(*mutex); +} + +static inline int pthread_mutex_unlock(pthread_mutex_t *mutex) +{ + return LWP_MutexUnlock(*mutex); +} + +static inline int pthread_join(pthread_t thread, void **retval) +{ + // FIXME: Shouldn't the second arg to LWP_JoinThread take retval? + (void)retval; + return LWP_JoinThread(thread, NULL); +} + +static inline int pthread_mutex_trylock(pthread_mutex_t *mutex) +{ + return LWP_MutexTryLock(*mutex); +} + +static inline int pthread_cond_wait(pthread_cond_t *cond, pthread_mutex_t *mutex) +{ + return LWP_CondWait(*cond, *mutex); +} + +static inline int pthread_cond_timedwait(pthread_cond_t *cond, pthread_mutex_t *mutex, const struct timespec *abstime) +{ + return LWP_CondWait(*cond, *mutex); +} + +static inline int pthread_cond_init(pthread_cond_t *cond, const pthread_condattr_t *attr) +{ + return LWP_CondInit(cond); +} + +static inline int pthread_cond_signal(pthread_cond_t *cond) +{ + return LWP_CondSignal(*cond); +} + +static inline int pthread_cond_destroy(pthread_cond_t *cond) +{ + return LWP_CondDestroy(*cond); +} + +#endif diff --git a/mednafen/src/FileWrapper.cpp b/mednafen/src/FileWrapper.cpp index b99d321512..792f71a158 100644 --- a/mednafen/src/FileWrapper.cpp +++ b/mednafen/src/FileWrapper.cpp @@ -24,7 +24,11 @@ #include #include //#include //zero 07-feb-2012 + +#ifdef _MSC_VER #include +#endif + #include //TODO - change FOP crap to separate functions. this was just less work at the time. diff --git a/mednafen/src/PSFLoader.cpp b/mednafen/src/PSFLoader.cpp index c2bf2c40b9..d41ee2d4e0 100644 --- a/mednafen/src/PSFLoader.cpp +++ b/mednafen/src/PSFLoader.cpp @@ -22,7 +22,7 @@ */ #include "mednafen.h" #include "PSFLoader.h" -#include "endian.h" +#include "mednafen-endian.h" #include "general.h" #include diff --git a/mednafen/src/cdrom/CDAccess_Image.cpp b/mednafen/src/cdrom/CDAccess_Image.cpp index f55c2313cb..6fc72ae3b8 100644 --- a/mednafen/src/cdrom/CDAccess_Image.cpp +++ b/mednafen/src/cdrom/CDAccess_Image.cpp @@ -42,7 +42,7 @@ #include #include "../general.h" -#include "../endian.h" +#include "../mednafen-endian.h" #include "../FileStream.h" #include "../MemoryStream.h" diff --git a/mednafen/src/cdrom/audioreader.cpp b/mednafen/src/cdrom/audioreader.cpp index 0d0b2c3bd3..e45b5bf056 100644 --- a/mednafen/src/cdrom/audioreader.cpp +++ b/mednafen/src/cdrom/audioreader.cpp @@ -46,7 +46,7 @@ #include #include "../general.h" -#include "../endian.h" +#include "../mednafen-endian.h" AudioReader::AudioReader() : LastReadPos(0) { @@ -579,4 +579,4 @@ AudioReader *AR_Open(Stream *fp) } -*/ \ No newline at end of file +*/ diff --git a/mednafen/src/cdrom/dvdisaster.h b/mednafen/src/cdrom/dvdisaster.h index c95af652e5..fb398c57da 100644 --- a/mednafen/src/cdrom/dvdisaster.h +++ b/mednafen/src/cdrom/dvdisaster.h @@ -35,7 +35,7 @@ *** Everything #includeable is rolled up herein... */ -#include "../types.h" +#include "../mednafen-types.h" #include #include diff --git a/mednafen/src/driver.h b/mednafen/src/driver.h index a295790570..30b5e063e1 100644 --- a/mednafen/src/driver.h +++ b/mednafen/src/driver.h @@ -1,7 +1,7 @@ #ifndef _MDFN_DRIVERH #define _MDFN_DRIVERH -#include "types.h" +#include "mednafen-types.h" #include // REMOVE ME eventually #include "git.h" #include "settings-driver.h" diff --git a/mednafen/src/drivers/args.cpp b/mednafen/src/drivers/args.cpp index 46ca9d43a8..42de076bd4 100644 --- a/mednafen/src/drivers/args.cpp +++ b/mednafen/src/drivers/args.cpp @@ -23,7 +23,7 @@ /* */ /****************************************************************/ -#include "../types.h" +#include "../mednafen-types.h" #include #include diff --git a/mednafen/src/endian.cpp b/mednafen/src/endian.cpp index ab5abca573..a4d415f0ab 100644 --- a/mednafen/src/endian.cpp +++ b/mednafen/src/endian.cpp @@ -16,7 +16,7 @@ */ #include "mednafen.h" -#include "endian.h" +#include "mednafen-endian.h" void Endian_A16_Swap(void *src, uint32 nelements) { diff --git a/mednafen/src/md5.cpp b/mednafen/src/md5.cpp index b7ea74872f..8b168b0d31 100644 --- a/mednafen/src/md5.cpp +++ b/mednafen/src/md5.cpp @@ -5,7 +5,7 @@ */ /* Converted to C++ for use in Mednafen */ -#include "types.h" +#include "mednafen-types.h" #include #include "md5.h" diff --git a/mednafen/src/endian.h b/mednafen/src/mednafen-endian.h similarity index 94% rename from mednafen/src/endian.h rename to mednafen/src/mednafen-endian.h index 17d9f32a55..9db2326f5f 100644 --- a/mednafen/src/endian.h +++ b/mednafen/src/mednafen-endian.h @@ -204,4 +204,4 @@ static INLINE uint32 MDFN_de32msb(const uint8 *morp) return(morp[3]|(morp[2]<<8)|(morp[1]<<16)|(morp[0]<<24)); } -#endif +#endif /* __MDFN_ENDIAN_H */ diff --git a/mednafen/src/memory.h b/mednafen/src/mednafen-memory.h similarity index 100% rename from mednafen/src/memory.h rename to mednafen/src/mednafen-memory.h diff --git a/mednafen/src/types.h b/mednafen/src/mednafen-types.h similarity index 100% rename from mednafen/src/types.h rename to mednafen/src/mednafen-types.h diff --git a/mednafen/src/mednafen.h b/mednafen/src/mednafen.h index 357ef58d5b..42b3e99a16 100644 --- a/mednafen/src/mednafen.h +++ b/mednafen/src/mednafen.h @@ -1,6 +1,7 @@ #ifndef _MEDNAFEN_H +#define _MEDNAFEN_H -#include "types.h" +#include "mednafen-types.h" #include #include #include @@ -53,8 +54,7 @@ int MDFN_RawInputStateAction(StateMem *sm, int load, int data_only); #include "mednafen-driver.h" -#include "endian.h" -#include "memory.h" +#include "mednafen-endian.h" +#include "mednafen-memory.h" -#define _MEDNAFEN_H -#endif +#endif /* _MEDNAFEN_H */ diff --git a/mednafen/src/memory.cpp b/mednafen/src/memory.cpp index 1e8c5163cb..96db0be5fa 100644 --- a/mednafen/src/memory.cpp +++ b/mednafen/src/memory.cpp @@ -20,7 +20,7 @@ #include #include -#include "memory.h" +#include "mednafen-memory.h" void *MDFN_calloc_real(uint32 nmemb, uint32 size, const char *purpose, const char *_file, const int _line) { diff --git a/mednafen/src/pcfx/jrevdct.cpp b/mednafen/src/pcfx/jrevdct.cpp index 2b62024960..cc0ccdb8e5 100644 --- a/mednafen/src/pcfx/jrevdct.cpp +++ b/mednafen/src/pcfx/jrevdct.cpp @@ -20,7 +20,7 @@ /* Modified 2007 for usage in Mednafen */ -#include "../types.h" +#include "../mednafen-types.h" #include "jrevdct.h" /* diff --git a/mednafen/src/psx/cpu.cpp b/mednafen/src/psx/cpu.cpp index 3f79d47b89..26db72186c 100644 --- a/mednafen/src/psx/cpu.cpp +++ b/mednafen/src/psx/cpu.cpp @@ -17,7 +17,10 @@ #include "psx.h" #include "cpu.h" + +#ifdef _MSC_VER #include +#endif //#define DPRINT(...) #define NOPRINT(...) diff --git a/mednafen/src/qtrecord.cpp b/mednafen/src/qtrecord.cpp index 0fa53ee56b..cd4b900c8e 100644 --- a/mednafen/src/qtrecord.cpp +++ b/mednafen/src/qtrecord.cpp @@ -16,7 +16,7 @@ */ #include "mednafen.h" -#include "endian.h" +#include "mednafen-endian.h" #include "qtrecord.h" #include "compress/minilzo.h" #include "video/png.h" diff --git a/mednafen/src/string/ConvertUTF.cpp b/mednafen/src/string/ConvertUTF.cpp index 3c06547b2f..24cfaabea2 100644 --- a/mednafen/src/string/ConvertUTF.cpp +++ b/mednafen/src/string/ConvertUTF.cpp @@ -39,7 +39,7 @@ ------------------------------------------------------------------------ */ -#include "../types.h" +#include "../mednafen-types.h" #include "ConvertUTF.h" #ifdef CVTUTF_DEBUG #include diff --git a/mednafen/src/video/png.cpp b/mednafen/src/video/png.cpp index 16fdcda0c9..647bd3cb40 100644 --- a/mednafen/src/video/png.cpp +++ b/mednafen/src/video/png.cpp @@ -19,7 +19,7 @@ #include #include "png.h" -#include "../endian.h" +#include "../mednafen-endian.h" void PNGWrite::WriteChunk(FileWrapper &pngfile, uint32 size, const char *type, const uint8 *data) { diff --git a/mednafen/src/wswan/debug.cpp b/mednafen/src/wswan/debug.cpp index 459021b3d3..c63cccc2b8 100644 --- a/mednafen/src/wswan/debug.cpp +++ b/mednafen/src/wswan/debug.cpp @@ -22,7 +22,7 @@ #include "v30mz.h" #include "debug.h" #include "dis/disasm.h" -#include "memory.h" +#include "mednafen-memory.h" #include "gfx.h" #include #include diff --git a/mednafen/src/wswan/dis/disasm.h b/mednafen/src/wswan/dis/disasm.h index b7da6960ba..f88bb47988 100644 --- a/mednafen/src/wswan/dis/disasm.h +++ b/mednafen/src/wswan/dis/disasm.h @@ -1,7 +1,7 @@ #ifndef _BX_DISASM_H_ #define _BX_DISASM_H_ -#include "../../types.h" +#include "../../mednafen-types.h" typedef uint16 bx_address; diff --git a/mednafen/src/wswan/v30mz.cpp b/mednafen/src/wswan/v30mz.cpp index f76661642d..43bfec9a57 100644 --- a/mednafen/src/wswan/v30mz.cpp +++ b/mednafen/src/wswan/v30mz.cpp @@ -28,7 +28,7 @@ */ #include "wswan.h" -#include "memory.h" +#include "mednafen-memory.h" #include