LIBRETRO: refactor OSystem_libretro

This commit is contained in:
Giovanni Cascione 2023-05-13 20:47:31 +02:00
parent da06ecad9b
commit 14335f50aa
13 changed files with 1461 additions and 1421 deletions

View File

@ -118,10 +118,14 @@ endif
INCLUDES += -I$(ROOT_PATH)/include
MODULE_PATHS += $(CORE_PATH)
LIBRETRO_OBJS := $(CORE_PATH)/libretro-os.o \
LIBRETRO_OBJS := $(CORE_PATH)/libretro.o \
$(CORE_PATH)/libretro-fs.o \
$(CORE_PATH)/libretro-fs-factory.o \
$(CORE_PATH)/libretro.o \
$(CORE_PATH)/libretro-os-base.o \
$(CORE_PATH)/libretro-os-graphics.o \
$(CORE_PATH)/libretro-os-events.o \
$(CORE_PATH)/libretro-os-inputs.o \
$(CORE_PATH)/libretro-os-utils.o \
$(CORE_PATH)/libretro-threads.o \
$(CORE_PATH)/libretro-timer.o

View File

@ -1,8 +1,4 @@
/* ScummVM - Graphic Adventure Engine
*
* ScummVM is the legal property of its developers, whose names
* are too numerous to list here. Please refer to the COPYRIGHT
* file distributed with this source distribution.
/* Copyright (C) 2023 Giovanni Cascione <ing.cascione@gmail.com>
*
* 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
@ -19,8 +15,8 @@
*
*/
#ifndef BACKENDS_LIBRETRO_OS_H
#define BACKENDS_LIBRETRO_OS_H
#ifndef LIBRETRO_DEFS_H
#define LIBRETRO_DEFS_H
#define SAMPLE_RATE 48000
#define REFRESH_RATE 60
@ -55,12 +51,6 @@
#define TEST_GAME_KO_NOT_FOUND 3
#define TEST_GAME_KO_MULTIPLE_RESULTS 4
#define FORBIDDEN_SYMBOL_ALLOW_ALL
#include "libretro.h"
#include "base/main.h"
#include "common/system.h"
#ifndef F_OK
#define F_OK 0
#endif
@ -73,25 +63,4 @@
#define R_OK 4
#endif
extern char cmd_params[20][200];
extern char cmd_params_num;
#if (defined(GEKKO) && !defined(WIIU)) || defined(__CELLOS_LV2__)
extern int access(const char *path, int amode);
#endif
OSystem *retroBuildOS();
const Graphics::Surface &getScreen();
void retroProcessMouse(retro_input_state_t aCallback, int device, float gamepad_cursor_speed, float gamepad_acceleration_time, bool analog_response_is_quadratic, int analog_deadzone, float mouse_speed);
void retroQuit(void);
void retroReset(void);
int retroTestGame(const char *game_id, bool autodetect);
void retroKeyEvent(bool down, unsigned keycode, uint32_t character, uint16_t key_modifiers);
uint8 getThreadSwitchCaller(void);
void retroDestroy(void);
#endif

View File

@ -0,0 +1,184 @@
/* Copyright (C) 2023 Giovanni Cascione <ing.cascione@gmail.com>
*
* 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 3 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, see <http://www.gnu.org/licenses/>.
*
*/
#ifndef BACKENDS_LIBRETRO_OS_H
#define BACKENDS_LIBRETRO_OS_H
#include <libretro.h>
#include <retro_miscellaneous.h>
#include "audio/mixer_intern.h"
#include "base/main.h"
#include "backends/base-backend.h"
#include "common/system.h"
#include "common/mutex.h"
#include "common/list.h"
#include "common/events.h"
#include "graphics/palette.h"
#include "graphics/surface.h"
#define LIBRETRO_G_SYSTEM dynamic_cast<OSystem_libretro *>(g_system)
extern retro_log_printf_t log_cb;
extern bool timing_inaccuracies_is_enabled(void);
extern bool consecutive_screen_updates_is_enabled(void);
extern void reset_performance_tuner(void);
extern void retro_osd_notification(const char* msg);
extern float frame_rate;
extern const char * retro_get_system_dir(void);
extern const char * retro_get_save_dir(void);
/**
* Dummy mutex implementation
*/
class LibretroMutexInternal final : public Common::MutexInternal {
public:
LibretroMutexInternal() {};
~LibretroMutexInternal() override {};
bool lock() override { return 0; }
bool unlock() override { return 0; };
};
class LibretroPalette {
public:
unsigned char _colors[256 * 3];
LibretroPalette(void);
~LibretroPalette(void) {};
void set(const byte *colors, uint start, uint num);
void get(byte *colors, uint start, uint num) const;
unsigned char *getColor(uint aIndex) const;
};
class OSystem_libretro : public EventsBaseBackend, public PaletteManager {
private:
int _mouseX;
int _mouseY;
int _relMouseX;
int _relMouseY;
int _mouseHotspotX;
int _mouseHotspotY;
int _mouseKeyColor;
float _mouseXAcc;
float _mouseYAcc;
float _dpadXAcc;
float _dpadYAcc;
float _dpadXVel;
float _dpadYVel;
unsigned _joypadnumpadLast;
uint32 _startTime;
uint8 _threadSwitchCaller;
Common::String s_systemDir;
Common::String s_saveDir;
Common::String s_extraDir;
Common::String s_themeDir;
Common::String s_lastDir;
static Common::List<Common::Event> _events;
public:
Audio::MixerImpl *_mixer;
Graphics::Surface _screen;
Graphics::Surface _gameScreen;
Graphics::Surface _overlay;
Graphics::Surface _mouseImage;
LibretroPalette _mousePalette;
LibretroPalette _gamePalette;
bool _overlayVisible;
bool _overlayInGUI;
bool _mouseDontScale;
bool _mouseButtons[2];
bool _joypadmouseButtons[2];
bool _joypadkeyboardButtons[8];
bool _joypadnumpadActive;
bool _ptrmouseButton;
bool _mousePaletteEnabled;
bool _mouseVisible;
/* Base */
OSystem_libretro(void);
~OSystem_libretro(void) override;
void initBackend(void) override;
void engineInit(void) override;
void engineDone(void) override;
bool hasFeature(Feature f) override;
void setFeatureState(Feature f, bool enable) override;
bool getFeatureState(Feature f) override;
void destroy(void);
void quit() override {}
/* Graphics */
Common::List<Graphics::PixelFormat> getSupportedFormats() const override;
const GraphicsMode *getSupportedGraphicsModes(void) const override;
void initSize(uint width, uint height, const Graphics::PixelFormat *format) override;
int16 getHeight(void) override;
int16 getWidth(void) override;
Graphics::PixelFormat getScreenFormat(void) const override;
void copyRectToScreen(const void *buf, int pitch, int x, int y, int w, int h) override;
void updateScreen(void) override;
void showOverlay(bool inGUI) override;
void hideOverlay(void) override;
void clearOverlay(void) override;
void grabOverlay(Graphics::Surface &surface) override;
void copyRectToOverlay(const void *buf, int pitch, int x, int y, int w, int h) override;
int16 getOverlayHeight(void) override;
int16 getOverlayWidth(void) override;
Graphics::PixelFormat getOverlayFormat() const override;
const Graphics::Surface &getScreen(void);
bool showMouse(bool visible) override;
void warpMouse(int x, int y) override;
void setMouseCursor(const void *buf, uint w, uint h, int hotspotX, int hotspotY, uint32 keycolor = 255, bool dontScale = false, const Graphics::PixelFormat *format = NULL, const byte *mask = nullptr) override;
void setCursorPalette(const byte *colors, uint start, uint num) override;
int getDefaultGraphicsMode() const override { return 0; }
bool isOverlayVisible() const override { return false; }
bool setGraphicsMode(int mode, uint flags = kGfxModeNoFlags) override { return true; }
int getGraphicsMode() const override { return 0; }
PaletteManager *getPaletteManager() override { return this; }
Graphics::Surface *lockScreen() override { return &_gameScreen; }
void unlockScreen() override {}
protected:
void setPalette(const byte *colors, uint start, uint num) override;
void grabPalette(byte *colors, uint start, uint num) const override;
/* Events */
public:
bool pollEvent(Common::Event &event) override;
uint8 getThreadSwitchCaller(void);
uint32 getMillis(bool skipRecord = false) override;
void delayMillis(uint msecs) override;
Common::MutexInternal *createMutex(void) override;
void requestQuit(void);
void resetQuit(void);
/* Utils */
void getTimeAndDate(TimeDate &t, bool skipRecord) const override;
Audio::Mixer *getMixer(void) override;
Common::String getDefaultConfigFileName(void) override;
void logMessage(LogMessageType::Type type, const char *message) override;
int testGame(const char *filedata, bool autodetect);
void addSysArchivesToSearchSet(Common::SearchSet &s, int priority = 0) override {}
private:
bool parseGameName(const Common::String &gameName, Common::String &engineId, Common::String &gameId);
/* Inputs */
public:
void processMouse(retro_input_state_t aCallback, int device, float gampad_cursor_speed, float gamepad_acceleration_time, bool analog_response_is_quadratic, int analog_deadzone, float mouse_speed);
static void processKeyEvent(bool down, unsigned keycode, uint32_t character, uint16_t key_modifiers);
void setShakePos(int shakeXOffset, int shakeYOffset) override {}
private:
void updateMouseXY(float deltaAcc, float * cumulativeXYAcc, int doing_x);
};
#endif

View File

@ -1,4 +1,4 @@
/* Copyright (C) 2022 Giovanni Cascione <ing.cascione@gmail.com>
/* Copyright (C) 2023 Giovanni Cascione <ing.cascione@gmail.com>
*
* 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
@ -21,7 +21,7 @@
#define EMU_THREAD_MIN_TIME 10
#include "backends/timer/default/default-timer.h"
#include "backends/platform/libretro/include/os.h"
#include "backends/platform/libretro/include/libretro-defs.h"
class LibretroTimerManager : public DefaultTimerManager {
uint32 _interval;

View File

@ -0,0 +1,126 @@
/* Copyright (C) 2023 Giovanni Cascione <ing.cascione@gmail.com>
*
* 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 3 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, see <http://www.gnu.org/licenses/>.
*
*/
#if defined(_WIN32)
#include "backends/fs/windows/windows-fs-factory.h"
#define FS_SYSTEM_FACTORY WindowsFilesystemFactory
#else
#include "backends/platform/libretro/include/libretro-fs-factory.h"
#define FS_SYSTEM_FACTORY LibRetroFilesystemFactory
#endif
#include "audio/mixer_intern.h"
#include "backends/base-backend.h"
#include "common/config-manager.h"
#include "common/system.h"
#include "graphics/surface.h"
#include "backends/saves/default/default-saves.h"
#include "backends/platform/libretro/include/libretro-timer.h"
#include "backends/platform/libretro/include/libretro-os.h"
#include "backends/platform/libretro/include/libretro-defs.h"
OSystem_libretro::OSystem_libretro() : _mousePaletteEnabled(false), _mouseVisible(false), _mouseX(0), _mouseY(0), _mouseXAcc(0.0), _mouseYAcc(0.0), _mouseHotspotX(0), _mouseHotspotY(0), _dpadXAcc(0.0), _dpadYAcc(0.0), _dpadXVel(0.0f), _dpadYVel(0.0f), _mouseKeyColor(0), _mouseDontScale(false), _joypadnumpadLast(8), _joypadnumpadActive(false), _mixer(0), _startTime(0), _threadSwitchCaller(0) {
_fsFactory = new FS_SYSTEM_FACTORY();
memset(_mouseButtons, 0, sizeof(_mouseButtons));
memset(_joypadmouseButtons, 0, sizeof(_joypadmouseButtons));
memset(_joypadkeyboardButtons, 0, sizeof(_joypadkeyboardButtons));
s_systemDir = Common::String(retro_get_system_dir());
s_saveDir = Common::String(retro_get_save_dir());
s_themeDir = s_systemDir + "/" + SCUMMVM_SYSTEM_SUBDIR + "/" + SCUMMVM_THEME_SUBDIR;
s_extraDir = s_systemDir + "/" + SCUMMVM_SYSTEM_SUBDIR + "/" + SCUMMVM_EXTRA_SUBDIR;
s_lastDir = s_systemDir;
_startTime = getMillis();
}
OSystem_libretro::~OSystem_libretro() {
_gameScreen.free();
_overlay.free();
_mouseImage.free();
_screen.free();
delete _mixer;
}
void OSystem_libretro::initBackend() {
_savefileManager = new DefaultSaveFileManager(s_saveDir);
if (! ConfMan.hasKey("themepath")) {
if (! Common::FSNode(s_themeDir).exists())
retro_osd_notification("ScummVM theme folder not found.");
else
ConfMan.set("themepath", s_themeDir);
}
if (! ConfMan.hasKey("extrapath")) {
if (! Common::FSNode(s_extraDir).exists())
retro_osd_notification("ScummVM datafiles folder not found. Some engines/features will not work.");
else
ConfMan.set("extrapath", s_extraDir);
}
if (! ConfMan.hasKey("browser_lastpath"))
ConfMan.set("browser_lastpath", s_lastDir);
#ifdef FRONTEND_SUPPORTS_RGB565
_overlay.create(RES_W_OVERLAY, RES_H_OVERLAY, Graphics::PixelFormat(2, 5, 6, 5, 0, 11, 5, 0, 0));
#else
_overlay.create(RES_W_OVERLAY, RES_H_OVERLAY, Graphics::PixelFormat(2, 5, 5, 5, 1, 10, 5, 0, 15));
#endif
_mixer = new Audio::MixerImpl(SAMPLE_RATE);
_timerManager = new LibretroTimerManager(frame_rate);
_mixer->setReady(true);
EventsBaseBackend::initBackend();
}
void OSystem_libretro::engineInit() {
Common::String engineId = ConfMan.get("engineid");
if (engineId.equalsIgnoreCase("scumm") && ConfMan.getBool("original_gui")) {
ConfMan.setBool("original_gui", false);
log_cb(RETRO_LOG_INFO, "\"original_gui\" setting forced to false\n");
}
}
void OSystem_libretro::engineDone() {
reset_performance_tuner();
}
bool OSystem_libretro::hasFeature(Feature f) {
return (f == OSystem::kFeatureCursorPalette);
}
void OSystem_libretro::setFeatureState(Feature f, bool enable) {
if (f == kFeatureCursorPalette)
_mousePaletteEnabled = enable;
}
bool OSystem_libretro::getFeatureState(Feature f) {
return (f == kFeatureCursorPalette) ? _mousePaletteEnabled : false;
}
Audio::Mixer *OSystem_libretro::getMixer() {
return _mixer;
}
void OSystem_libretro::destroy() {
delete this;
}

View File

@ -0,0 +1,116 @@
/* Copyright (C) 2023 Giovanni Cascione <ing.cascione@gmail.com>
*
* 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 3 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, see <http://www.gnu.org/licenses/>.
*
*/
#include <unistd.h>
#include <features/features_cpu.h>
#include <sys/time.h>
#if defined(__CELLOS_LV2__)
#include <sys/sys_time.h>
#elif (defined(GEKKO) && !defined(WIIU))
#include <ogc/lwp_watchdog.h>
#else
#include <time.h>
#endif
#include "common/list.h"
#include "common/events.h"
#include "backends/platform/libretro/include/libretro-os.h"
#include "backends/platform/libretro/include/libretro-timer.h"
#include "backends/platform/libretro/include/libretro-defs.h"
Common::List<Common::Event> OSystem_libretro::_events;
bool OSystem_libretro::pollEvent(Common::Event &event) {
_threadSwitchCaller = THREAD_SWITCH_POLL;
((LibretroTimerManager *)_timerManager)->checkThread();
((LibretroTimerManager *)_timerManager)->handler();
if (!_events.empty()) {
event = _events.front();
_events.pop_front();
return true;
}
return false;
}
uint8 OSystem_libretro::getThreadSwitchCaller(){
return _threadSwitchCaller;
}
uint32 OSystem_libretro::getMillis(bool skipRecord) {
#if (defined(GEKKO) && !defined(WIIU))
return (ticks_to_microsecs(gettime()) / 1000.0) - _startTime;
#elif defined(WIIU)
return ((cpu_features_get_time_usec()) / 1000) - _startTime;
#elif defined(__CELLOS_LV2__)
return (sys_time_get_system_time() / 1000.0) - _startTime;
#else
struct timeval t;
gettimeofday(&t, 0);
return ((t.tv_sec * 1000) + (t.tv_usec / 1000)) - _startTime;
#endif
}
void OSystem_libretro::delayMillis(uint msecs) {
uint32 start_time = getMillis();
uint32 elapsed_time = 0;
_threadSwitchCaller = THREAD_SWITCH_DELAY;
if (timing_inaccuracies_is_enabled()) {
while (elapsed_time < msecs) {
/* When remaining delay would take us past the next thread switch time, we switch immediately
in order to burn as much as possible delay time in the main RetroArch thread as soon as possible. */
if (msecs - elapsed_time >= ((LibretroTimerManager *)_timerManager)->timeToNextSwitch())
((LibretroTimerManager *)_timerManager)->switchThread();
else
usleep(1000);
/* Actual delay provided will be lower than requested: elapsed time is calculated cumulatively.
i.e. the higher the requested delay, the higher the actual delay reduction */
elapsed_time += getMillis() - start_time;
}
} else {
while (elapsed_time < msecs) {
/* if remaining delay is lower than last amount of time spent on main thread, burn it in emu thread
to avoid exceeding requested delay */
if (msecs - elapsed_time >= ((LibretroTimerManager *)_timerManager)->spentOnMainThread() && !((LibretroTimerManager *)_timerManager)->timeToNextSwitch())
((LibretroTimerManager *)_timerManager)->switchThread();
else
usleep(1000);
elapsed_time = getMillis() - start_time;
}
}
((LibretroTimerManager *)_timerManager)->handler();
}
Common::MutexInternal *OSystem_libretro::createMutex(void) {
return new LibretroMutexInternal();
}
void OSystem_libretro::requestQuit() {
Common::Event ev;
ev.type = Common::EVENT_QUIT;
LIBRETRO_G_SYSTEM->getEventManager()->pushEvent(ev);
}
void OSystem_libretro::resetQuit() {
LIBRETRO_G_SYSTEM->getEventManager()->resetQuit();
}

View File

@ -0,0 +1,411 @@
/* Copyright (C) 2023 Giovanni Cascione <ing.cascione@gmail.com>
*
* 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 3 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, see <http://www.gnu.org/licenses/>.
*
*/
#include <retro_inline.h>
//#include "common/system.h"
#include "graphics/colormasks.h"
#include "graphics/palette.h"
#include "graphics/surface.h"
#include "backends/platform/libretro/include/libretro-os.h"
#include "backends/platform/libretro/include/libretro-timer.h"
#include "backends/platform/libretro/include/libretro-defs.h"
static INLINE void blit_uint8_uint16_fast(Graphics::Surface &aOut, const Graphics::Surface &aIn, const LibretroPalette &aColors) {
for (int i = 0; i < aIn.h; i++) {
if (i >= aOut.h)
continue;
uint8_t *const in = (uint8_t *)aIn.getPixels() + (i * aIn.w);
uint16_t *const out = (uint16_t *)aOut.getPixels() + (i * aOut.w);
for (int j = 0; j < aIn.w; j++) {
if (j >= aOut.w)
continue;
uint8 r, g, b;
const uint8_t val = in[j];
// if(val != 0xFFFFFFFF)
{
if (aIn.format.bytesPerPixel == 1) {
unsigned char *col = aColors.getColor(val);
r = *col++;
g = *col++;
b = *col++;
} else
aIn.format.colorToRGB(in[j], r, g, b);
out[j] = aOut.format.RGBToColor(r, g, b);
}
}
}
}
static INLINE void blit_uint32_uint16(Graphics::Surface &aOut, const Graphics::Surface &aIn, const LibretroPalette &aColors) {
for (int i = 0; i < aIn.h; i++) {
if (i >= aOut.h)
continue;
uint32_t *const in = (uint32_t *)aIn.getPixels() + (i * aIn.w);
uint16_t *const out = (uint16_t *)aOut.getPixels() + (i * aOut.w);
for (int j = 0; j < aIn.w; j++) {
if (j >= aOut.w)
continue;
uint8 r, g, b;
// const uint32_t val = in[j];
// if(val != 0xFFFFFFFF)
{
aIn.format.colorToRGB(in[j], r, g, b);
out[j] = aOut.format.RGBToColor(r, g, b);
}
}
}
}
static INLINE void blit_uint16_uint16(Graphics::Surface &aOut, const Graphics::Surface &aIn, const LibretroPalette &aColors) {
for (int i = 0; i < aIn.h; i++) {
if (i >= aOut.h)
continue;
uint16_t *const in = (uint16_t *)aIn.getPixels() + (i * aIn.w);
uint16_t *const out = (uint16_t *)aOut.getPixels() + (i * aOut.w);
for (int j = 0; j < aIn.w; j++) {
if (j >= aOut.w)
continue;
uint8 r, g, b;
// const uint16_t val = in[j];
// if(val != 0xFFFFFFFF)
{
aIn.format.colorToRGB(in[j], r, g, b);
out[j] = aOut.format.RGBToColor(r, g, b);
}
}
}
}
static void blit_uint8_uint16(Graphics::Surface &aOut, const Graphics::Surface &aIn, int aX, int aY, const LibretroPalette &aColors, uint32 aKeyColor) {
for (int i = 0; i < aIn.h; i++) {
if ((i + aY) < 0 || (i + aY) >= aOut.h)
continue;
uint8_t *const in = (uint8_t *)aIn.getPixels() + (i * aIn.w);
uint16_t *const out = (uint16_t *)aOut.getPixels() + ((i + aY) * aOut.w);
for (int j = 0; j < aIn.w; j++) {
if ((j + aX) < 0 || (j + aX) >= aOut.w)
continue;
uint8 r, g, b;
const uint8_t val = in[j];
if (val != aKeyColor) {
unsigned char *col = aColors.getColor(val);
r = *col++;
g = *col++;
b = *col++;
out[j + aX] = aOut.format.RGBToColor(r, g, b);
}
}
}
}
static void blit_uint16_uint16(Graphics::Surface &aOut, const Graphics::Surface &aIn, int aX, int aY, const LibretroPalette &aColors, uint32 aKeyColor) {
for (int i = 0; i < aIn.h; i++) {
if ((i + aY) < 0 || (i + aY) >= aOut.h)
continue;
uint16_t *const in = (uint16_t *)aIn.getPixels() + (i * aIn.w);
uint16_t *const out = (uint16_t *)aOut.getPixels() + ((i + aY) * aOut.w);
for (int j = 0; j < aIn.w; j++) {
if ((j + aX) < 0 || (j + aX) >= aOut.w)
continue;
uint8 r, g, b;
const uint16_t val = in[j];
if (val != aKeyColor) {
aIn.format.colorToRGB(in[j], r, g, b);
out[j + aX] = aOut.format.RGBToColor(r, g, b);
}
}
}
}
static void blit_uint32_uint16(Graphics::Surface &aOut, const Graphics::Surface &aIn, int aX, int aY, const LibretroPalette &aColors, uint32 aKeyColor) {
for (int i = 0; i < aIn.h; i++) {
if ((i + aY) < 0 || (i + aY) >= aOut.h)
continue;
uint32_t *const in = (uint32_t *)aIn.getPixels() + (i * aIn.w);
uint16_t *const out = (uint16_t *)aOut.getPixels() + ((i + aY) * aOut.w);
for (int j = 0; j < aIn.w; j++) {
if ((j + aX) < 0 || (j + aX) >= aOut.w)
continue;
uint8 in_a, in_r, in_g, in_b;
uint8 out_r, out_g, out_b;
uint32_t blend_r, blend_g, blend_b;
const uint32_t val = in[j];
if (val != aKeyColor) {
aIn.format.colorToARGB(in[j], in_a, in_r, in_g, in_b);
if (in_a) {
aOut.format.colorToRGB(out[j + aX], out_r, out_g, out_b);
blend_r = ((in_r * in_a) + (out_r * (255 - in_a))) / 255;
blend_g = ((in_g * in_a) + (out_g * (255 - in_a))) / 255;
blend_b = ((in_b * in_a) + (out_b * (255 - in_a))) / 255;
out[j + aX] = aOut.format.RGBToColor(blend_r, blend_g, blend_b);
}
}
}
}
}
static INLINE void copyRectToSurface(uint8_t *pixels, int out_pitch, const uint8_t *src, int pitch, int x, int y, int w, int h, int out_bpp) {
uint8_t *dst = pixels + y * out_pitch + x * out_bpp;
do {
memcpy(dst, src, w * out_bpp);
src += pitch;
dst += out_pitch;
} while (--h);
}
LibretroPalette::LibretroPalette() {
memset(_colors, 0, sizeof(_colors));
}
void LibretroPalette::set(const byte *colors, uint start, uint num) {
memcpy(_colors + start * 3, colors, num * 3);
}
void LibretroPalette::get(byte *colors, uint start, uint num) const {
memcpy(colors, _colors + start * 3, num * 3);
}
unsigned char * LibretroPalette::getColor(uint aIndex) const {
return (unsigned char *)&_colors[aIndex * 3];
}
Common::List<Graphics::PixelFormat> OSystem_libretro::getSupportedFormats() const {
Common::List<Graphics::PixelFormat> result;
#ifdef SCUMM_LITTLE_ENDIAN
/* RGBA8888 */
result.push_back(Graphics::PixelFormat(4, 8, 8, 8, 8, 24, 16, 8, 0));
#else
/* ABGR8888 */
result.push_back(Graphics::PixelFormat(4, 8, 8, 8, 8, 0, 8, 16, 24));
#endif
#ifdef FRONTEND_SUPPORTS_RGB565
/* RGB565 - overlay */
result.push_back(Graphics::PixelFormat(2, 5, 6, 5, 0, 11, 5, 0, 0));
#endif
/* RGB555 - fmtowns */
result.push_back(Graphics::PixelFormat(2, 5, 5, 5, 1, 10, 5, 0, 15));
/* Palette - most games */
result.push_back(Graphics::PixelFormat::createFormatCLUT8());
return result;
}
const OSystem_libretro::GraphicsMode *OSystem_libretro::getSupportedGraphicsModes() const {
static const OSystem::GraphicsMode s_noGraphicsModes[] = {{0, 0, 0}};
return s_noGraphicsModes;
}
void OSystem_libretro::initSize(uint width, uint height, const Graphics::PixelFormat *format) {
_gameScreen.create(width, height, format ? *format : Graphics::PixelFormat::createFormatCLUT8());
}
int16 OSystem_libretro::getHeight() {
return _gameScreen.h;
}
int16 OSystem_libretro::getWidth() {
return _gameScreen.w;
}
Graphics::PixelFormat OSystem_libretro::getScreenFormat() const {
return _gameScreen.format;
}
void OSystem_libretro::copyRectToScreen(const void *buf, int pitch, int x, int y, int w, int h) {
const uint8_t *src = (const uint8_t *)buf;
uint8_t *pix = (uint8_t *)_gameScreen.getPixels();
copyRectToSurface(pix, _gameScreen.pitch, src, pitch, x, y, w, h, _gameScreen.format.bytesPerPixel);
}
void OSystem_libretro::updateScreen() {
const Graphics::Surface &srcSurface = (_overlayInGUI) ? _overlay : _gameScreen;
if (srcSurface.w && srcSurface.h) {
switch (srcSurface.format.bytesPerPixel) {
case 1:
case 3:
blit_uint8_uint16_fast(_screen, srcSurface, _gamePalette);
break;
case 2:
blit_uint16_uint16(_screen, srcSurface, _gamePalette);
break;
case 4:
blit_uint32_uint16(_screen, srcSurface, _gamePalette);
break;
}
}
// Draw Mouse
if (_mouseVisible && _mouseImage.w && _mouseImage.h) {
const int x = _mouseX - _mouseHotspotX;
const int y = _mouseY - _mouseHotspotY;
switch (_mouseImage.format.bytesPerPixel) {
case 1:
case 3:
blit_uint8_uint16(_screen, _mouseImage, x, y, _mousePaletteEnabled ? _mousePalette : _gamePalette, _mouseKeyColor);
break;
case 2:
blit_uint16_uint16(_screen, _mouseImage, x, y, _mousePaletteEnabled ? _mousePalette : _gamePalette, _mouseKeyColor);
break;
case 4:
blit_uint32_uint16(_screen, _mouseImage, x, y, _mousePaletteEnabled ? _mousePalette : _gamePalette, _mouseKeyColor);
break;
}
}
/* Switch directly to main thread in case of consecutive updateScreen, to avoid losing frames.
Non consecutive updateScreen are covered by thread switches triggered by pollEvent or delayMillis. */
if (! timing_inaccuracies_is_enabled() && consecutive_screen_updates_is_enabled()) {
if (_threadSwitchCaller & THREAD_SWITCH_UPDATE) {
((LibretroTimerManager *)_timerManager)->switchThread();
} else {
_threadSwitchCaller = THREAD_SWITCH_UPDATE;
}
}
}
void OSystem_libretro::showOverlay(bool inGUI) {
_overlayVisible = true;
_overlayInGUI = inGUI;
}
void OSystem_libretro::hideOverlay() {
_overlayVisible = false;
_overlayInGUI = false;
}
void OSystem_libretro::clearOverlay() {
_overlay.fillRect(Common::Rect(_overlay.w, _overlay.h), 0);
}
void OSystem_libretro::grabOverlay(Graphics::Surface &surface) {
const unsigned char *src = (unsigned char *)_overlay.getPixels();
unsigned char *dst = (byte *)surface.getPixels();
;
unsigned i = RES_H_OVERLAY;
do {
memcpy(dst, src, RES_W_OVERLAY << 1);
dst += surface.pitch;
src += RES_W_OVERLAY << 1;
} while (--i);
}
void OSystem_libretro::copyRectToOverlay(const void *buf, int pitch, int x, int y, int w, int h) {
const uint8_t *src = (const uint8_t *)buf;
uint8_t *pix = (uint8_t *)_overlay.getPixels();
copyRectToSurface(pix, _overlay.pitch, src, pitch, x, y, w, h, _overlay.format.bytesPerPixel);
}
int16 OSystem_libretro::getOverlayHeight() {
return _overlay.h;
}
int16 OSystem_libretro::getOverlayWidth() {
return _overlay.w;
}
Graphics::PixelFormat OSystem_libretro::getOverlayFormat() const {
return _overlay.format;
}
bool OSystem_libretro::showMouse(bool visible) {
const bool wasVisible = _mouseVisible;
_mouseVisible = visible;
return wasVisible;
}
void OSystem_libretro::warpMouse(int x, int y) {
_mouseX = x;
_mouseY = y;
}
void OSystem_libretro::setMouseCursor(const void *buf, uint w, uint h, int hotspotX, int hotspotY, uint32 keycolor, bool dontScale, const Graphics::PixelFormat *format, const byte *mask) {
const Graphics::PixelFormat mformat = format ? *format : Graphics::PixelFormat::createFormatCLUT8();
if (_mouseImage.w != w || _mouseImage.h != h || _mouseImage.format != mformat) {
_mouseImage.create(w, h, mformat);
}
memcpy(_mouseImage.getPixels(), buf, h * _mouseImage.pitch);
_mouseHotspotX = hotspotX;
_mouseHotspotY = hotspotY;
_mouseKeyColor = keycolor;
_mouseDontScale = dontScale;
}
void OSystem_libretro::setCursorPalette(const byte *colors, uint start, uint num) {
_mousePalette.set(colors, start, num);
_mousePaletteEnabled = true;
}
const Graphics::Surface &OSystem_libretro::getScreen() {
const Graphics::Surface &srcSurface = (_overlayInGUI) ? _overlay : _gameScreen;
if (srcSurface.w != _screen.w || srcSurface.h != _screen.h) {
#ifdef FRONTEND_SUPPORTS_RGB565
_screen.create(srcSurface.w, srcSurface.h, Graphics::PixelFormat(2, 5, 6, 5, 0, 11, 5, 0, 0));
#else
_screen.create(srcSurface.w, srcSurface.h, Graphics::PixelFormat(2, 5, 5, 5, 1, 10, 5, 0, 15));
#endif
}
return _screen;
}
void OSystem_libretro::setPalette(const byte *colors, uint start, uint num) {
_gamePalette.set(colors, start, num);
}
void OSystem_libretro::grabPalette(byte *colors, uint start, uint num) const {
_gamePalette.get(colors, start, num);
}

View File

@ -0,0 +1,437 @@
/* Copyright (C) 2023 Giovanni Cascione <ing.cascione@gmail.com>
*
* 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 3 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, see <http://www.gnu.org/licenses/>.
*
*/
#include <libretro.h>
#include "backends/platform/libretro/include/libretro-os.h"
#define ANALOG_RANGE 0x8000
#define BASE_CURSOR_SPEED 4
#define PI 3.141592653589793238
void OSystem_libretro::updateMouseXY(float deltaAcc, float * cumulativeXYAcc, int doing_x){
int * mouseXY;
int16 * screen_wh;
int * relMouseXY;
int cumulativeXYAcc_int;
if (doing_x) {
mouseXY = &_mouseX;
screen_wh = &_screen.w;
relMouseXY = &_relMouseX;
} else {
mouseXY = &_mouseY;
screen_wh = &_screen.h;
relMouseXY = &_relMouseY;
}
*cumulativeXYAcc += deltaAcc;
cumulativeXYAcc_int = (int)*cumulativeXYAcc;
if (cumulativeXYAcc_int != 0) {
// Set mouse position
*mouseXY += cumulativeXYAcc_int;
*mouseXY = (*mouseXY < 0) ? 0 : *mouseXY;
*mouseXY = (*mouseXY >= *screen_wh) ? *screen_wh : *mouseXY;
// Update accumulator
*cumulativeXYAcc -= (float)cumulativeXYAcc_int;
}
*relMouseXY = (int)deltaAcc;
}
void OSystem_libretro::processMouse(retro_input_state_t aCallback, int device, float gampad_cursor_speed, float gamepad_acceleration_time, bool analog_response_is_quadratic, int analog_deadzone, float mouse_speed) {
enum processMouse_status {
STATUS_DOING_JOYSTICK = (1 << 0),
STATUS_DOING_MOUSE = (1 << 1),
STATUS_DOING_X = (1 << 2),
STATUS_DOING_Y = (1 << 3)
};
uint8_t status = 0;
int16_t joy_x, joy_y, joy_rx, joy_ry, x, y;
float analog_amplitude_x, analog_amplitude_y;
float deltaAcc;
bool down;
float screen_adjusted_cursor_speed = (float)_screen.w / 320.0f; // Dpad cursor speed should always be based off a 320 wide screen, to keep speeds consistent
float adjusted_cursor_speed = (float)BASE_CURSOR_SPEED * gampad_cursor_speed * screen_adjusted_cursor_speed;
float inverse_acceleration_time = (gamepad_acceleration_time > 0.0) ? (1.0 / 60.0) * (1.0 / gamepad_acceleration_time) : 1.0;
int dpad_cursor_offset;
double rs_radius, rs_angle;
unsigned numpad_index;
static const uint32_t retroButtons[2] = {RETRO_DEVICE_ID_MOUSE_LEFT, RETRO_DEVICE_ID_MOUSE_RIGHT};
static const Common::EventType eventID[2][2] = {{Common::EVENT_LBUTTONDOWN, Common::EVENT_LBUTTONUP}, {Common::EVENT_RBUTTONDOWN, Common::EVENT_RBUTTONUP}};
static const unsigned gampad_key_map[8][4] = {
{RETRO_DEVICE_ID_JOYPAD_X, (unsigned)Common::KEYCODE_ESCAPE, (unsigned)Common::ASCII_ESCAPE, 0}, // Esc
{RETRO_DEVICE_ID_JOYPAD_Y, (unsigned)Common::KEYCODE_PERIOD, 46, 0}, // .
{RETRO_DEVICE_ID_JOYPAD_L, (unsigned)Common::KEYCODE_RETURN, (unsigned)Common::ASCII_RETURN, 0}, // Enter
{RETRO_DEVICE_ID_JOYPAD_R, (unsigned)Common::KEYCODE_KP5, 53, 0}, // Numpad 5
{RETRO_DEVICE_ID_JOYPAD_L2, (unsigned)Common::KEYCODE_BACKSPACE, (unsigned)Common::ASCII_BACKSPACE, 0}, // Backspace
{RETRO_DEVICE_ID_JOYPAD_L3, (unsigned)Common::KEYCODE_F10, (unsigned)Common::ASCII_F10, 0}, // F10
{RETRO_DEVICE_ID_JOYPAD_R3, (unsigned)Common::KEYCODE_KP0, 48, 0}, // Numpad 0
{RETRO_DEVICE_ID_JOYPAD_SELECT, (unsigned)Common::KEYCODE_F7, (unsigned)Common::ASCII_F7, RETROKMOD_CTRL}, // CTRL+F7 (virtual keyboard)
};
// Right stick circular wrap around: 1 -> 2 -> 3 -> 6 -> 9 -> 8 -> 7 -> 4
static const unsigned gampad_numpad_map[8][2] = {
{(unsigned)Common::KEYCODE_KP1, 49},
{(unsigned)Common::KEYCODE_KP2, 50},
{(unsigned)Common::KEYCODE_KP3, 51},
{(unsigned)Common::KEYCODE_KP6, 54},
{(unsigned)Common::KEYCODE_KP9, 57},
{(unsigned)Common::KEYCODE_KP8, 56},
{(unsigned)Common::KEYCODE_KP7, 55},
{(unsigned)Common::KEYCODE_KP4, 52},
};
// Reduce gamepad cursor speed, if required
if (device == RETRO_DEVICE_JOYPAD && aCallback(0, RETRO_DEVICE_JOYPAD, 0, RETRO_DEVICE_ID_JOYPAD_R2)) {
adjusted_cursor_speed = adjusted_cursor_speed * (1.0f / 5.0f);
}
status = 0;
x = aCallback(0, RETRO_DEVICE_MOUSE, 0, RETRO_DEVICE_ID_MOUSE_X);
y = aCallback(0, RETRO_DEVICE_MOUSE, 0, RETRO_DEVICE_ID_MOUSE_Y);
joy_x = aCallback(0, RETRO_DEVICE_ANALOG, RETRO_DEVICE_INDEX_ANALOG_LEFT, RETRO_DEVICE_ID_ANALOG_X);
joy_y = aCallback(0, RETRO_DEVICE_ANALOG, RETRO_DEVICE_INDEX_ANALOG_LEFT, RETRO_DEVICE_ID_ANALOG_Y);
// Left Analog X Axis
if (joy_x > analog_deadzone || joy_x < -analog_deadzone) {
status |= (STATUS_DOING_JOYSTICK | STATUS_DOING_X);
if (joy_x > analog_deadzone) {
// Reset accumulator when changing direction
_mouseXAcc = (_mouseXAcc < 0.0) ? 0.0 : _mouseXAcc;
joy_x = joy_x - analog_deadzone;
}
if (joy_x < -analog_deadzone) {
// Reset accumulator when changing direction
_mouseXAcc = (_mouseXAcc > 0.0) ? 0.0 : _mouseXAcc;
joy_x = joy_x + analog_deadzone;
}
// Update accumulator
analog_amplitude_x = (float)joy_x / (float)(ANALOG_RANGE - analog_deadzone);
if (analog_response_is_quadratic) {
if (analog_amplitude_x < 0.0)
analog_amplitude_x = -(analog_amplitude_x * analog_amplitude_x);
else
analog_amplitude_x = analog_amplitude_x * analog_amplitude_x;
}
// printf("analog_amplitude_x: %f\n", analog_amplitude_x);
deltaAcc = analog_amplitude_x * adjusted_cursor_speed;
updateMouseXY(deltaAcc, &_mouseXAcc, 1);
}
// Left Analog Y Axis
if (joy_y > analog_deadzone || joy_y < -analog_deadzone) {
status |= (STATUS_DOING_JOYSTICK | STATUS_DOING_Y);
if (joy_y > analog_deadzone) {
// Reset accumulator when changing direction
_mouseYAcc = (_mouseYAcc < 0.0) ? 0.0 : _mouseYAcc;
joy_y = joy_y - analog_deadzone;
}
if (joy_y < -analog_deadzone) {
// Reset accumulator when changing direction
_mouseYAcc = (_mouseYAcc > 0.0) ? 0.0 : _mouseYAcc;
joy_y = joy_y + analog_deadzone;
}
// Update accumulator
analog_amplitude_y = (float)joy_y / (float)(ANALOG_RANGE - analog_deadzone);
if (analog_response_is_quadratic) {
if (analog_amplitude_y < 0.0)
analog_amplitude_y = -(analog_amplitude_y * analog_amplitude_y);
else
analog_amplitude_y = analog_amplitude_y * analog_amplitude_y;
}
// printf("analog_amplitude_y: %f\n", analog_amplitude_y);
deltaAcc = analog_amplitude_y * adjusted_cursor_speed;
updateMouseXY(deltaAcc, &_mouseYAcc, 0);
}
if (device == RETRO_DEVICE_JOYPAD) {
bool dpadLeft = aCallback(0, RETRO_DEVICE_JOYPAD, 0, RETRO_DEVICE_ID_JOYPAD_LEFT);
bool dpadRight = aCallback(0, RETRO_DEVICE_JOYPAD, 0, RETRO_DEVICE_ID_JOYPAD_RIGHT);
bool dpadUp = aCallback(0, RETRO_DEVICE_JOYPAD, 0, RETRO_DEVICE_ID_JOYPAD_UP);
bool dpadDown = aCallback(0, RETRO_DEVICE_JOYPAD, 0, RETRO_DEVICE_ID_JOYPAD_DOWN);
if (dpadLeft || dpadRight) {
status |= (STATUS_DOING_JOYSTICK | STATUS_DOING_X);
_dpadXVel = MIN(_dpadXVel + inverse_acceleration_time, 1.0f);
if (dpadLeft) {
deltaAcc = -(_dpadXVel * adjusted_cursor_speed);
_dpadXAcc = _dpadXAcc < deltaAcc ? _dpadXAcc : 0.0f;
} else { //dpadRight
deltaAcc = _dpadXVel * adjusted_cursor_speed;
_dpadXAcc = _dpadXAcc > deltaAcc ? _dpadXAcc : 0.0f;
}
updateMouseXY(deltaAcc, &_dpadXAcc, 1);
} else {
_dpadXVel = 0.0f;
}
if (dpadUp || dpadDown) {
status |= (STATUS_DOING_JOYSTICK | STATUS_DOING_Y);
_dpadYVel = MIN(_dpadYVel + inverse_acceleration_time, 1.0f);
if (dpadUp) {
deltaAcc = -(_dpadYVel * adjusted_cursor_speed);
_dpadYAcc = _dpadYAcc < deltaAcc ? _dpadYAcc : 0.0f;
} else { //dpadDown
deltaAcc = _dpadYVel * adjusted_cursor_speed;
_dpadYAcc = _dpadYAcc > deltaAcc ? _dpadYAcc : 0.0f;
}
updateMouseXY(deltaAcc, &_dpadYAcc, 0);
} else {
_dpadYVel = 0.0f;
}
if (aCallback(0, RETRO_DEVICE_JOYPAD, 0, RETRO_DEVICE_ID_JOYPAD_START)) {
Common::Event ev;
ev.type = Common::EVENT_MAINMENU;
_events.push_back(ev);
}
}
#if defined(WIIU) || defined(__SWITCH__)
int p_x = aCallback(0, RETRO_DEVICE_POINTER, 0, RETRO_DEVICE_ID_POINTER_X);
int p_y = aCallback(0, RETRO_DEVICE_POINTER, 0, RETRO_DEVICE_ID_POINTER_Y);
int p_press = aCallback(0, RETRO_DEVICE_POINTER, 0, RETRO_DEVICE_ID_POINTER_PRESSED);
int px = (int)((p_x + 0x7fff) * _screen.w / 0xffff);
int py = (int)((p_y + 0x7fff) * _screen.h / 0xffff);
// printf("(%d,%d) p:%d\n",px,py,pp);
static int ptrhold = 0;
if (p_press)
ptrhold++;
else
ptrhold = 0;
if (ptrhold > 0) {
_mouseX = px;
_mouseY = py;
Common::Event ev;
ev.type = Common::EVENT_MOUSEMOVE;
ev.mouse.x = _mouseX;
ev.mouse.y = _mouseY;
_events.push_back(ev);
}
if (ptrhold > 10 && _ptrmouseButton == 0) {
_ptrmouseButton = 1;
Common::Event ev;
ev.type = eventID[0][_ptrmouseButton ? 0 : 1];
ev.mouse.x = _mouseX;
ev.mouse.y = _mouseY;
_events.push_back(ev);
} else if (ptrhold == 0 && _ptrmouseButton == 1) {
_ptrmouseButton = 0;
Common::Event ev;
ev.type = eventID[0][_ptrmouseButton ? 0 : 1];
ev.mouse.x = _mouseX;
ev.mouse.y = _mouseY;
_events.push_back(ev);
}
#endif
if (status & STATUS_DOING_JOYSTICK) {
Common::Event ev;
ev.type = Common::EVENT_MOUSEMOVE;
ev.mouse.x = _mouseX;
ev.mouse.y = _mouseY;
ev.relMouse.x = status & STATUS_DOING_X ? _relMouseX : 0;
ev.relMouse.y = status & STATUS_DOING_Y ? _relMouseY : 0;
_events.push_back(ev);
}
// Gampad mouse buttons
down = aCallback(0, RETRO_DEVICE_JOYPAD, 0, RETRO_DEVICE_ID_JOYPAD_A);
if (down != _joypadmouseButtons[0]) {
_joypadmouseButtons[0] = down;
Common::Event ev;
ev.type = eventID[0][down ? 0 : 1];
ev.mouse.x = _mouseX;
ev.mouse.y = _mouseY;
_events.push_back(ev);
}
down = aCallback(0, RETRO_DEVICE_JOYPAD, 0, RETRO_DEVICE_ID_JOYPAD_B);
if (down != _joypadmouseButtons[1]) {
_joypadmouseButtons[1] = down;
Common::Event ev;
ev.type = eventID[1][down ? 0 : 1];
ev.mouse.x = _mouseX;
ev.mouse.y = _mouseY;
_events.push_back(ev);
}
// Gamepad keyboard buttons
for (int i = 0; i < 8; i++) {
down = aCallback(0, RETRO_DEVICE_JOYPAD, 0, gampad_key_map[i][0]);
if (down != _joypadkeyboardButtons[i]) {
_joypadkeyboardButtons[i] = down;
bool state = down ? true : false;
processKeyEvent(state, gampad_key_map[i][1], (uint32_t)gampad_key_map[i][2], (uint32_t)gampad_key_map[i][3]);
}
}
// Gamepad right stick numpad emulation
joy_rx = aCallback(0, RETRO_DEVICE_ANALOG, RETRO_DEVICE_INDEX_ANALOG_RIGHT, RETRO_DEVICE_ID_ANALOG_X);
joy_ry = aCallback(0, RETRO_DEVICE_ANALOG, RETRO_DEVICE_INDEX_ANALOG_RIGHT, RETRO_DEVICE_ID_ANALOG_Y);
if (joy_rx > analog_deadzone)
joy_rx = joy_rx - analog_deadzone;
else if (joy_rx < -analog_deadzone)
joy_rx = joy_rx + analog_deadzone;
else
joy_rx = 0;
if (joy_ry > analog_deadzone)
joy_ry = joy_ry - analog_deadzone;
else if (joy_ry < -analog_deadzone)
joy_ry = joy_ry + analog_deadzone;
else
joy_ry = 0;
// This is very ugly, but I don't have time to make it nicer...
if (joy_rx != 0 || joy_ry != 0) {
analog_amplitude_x = (float)joy_rx / (float)(ANALOG_RANGE - analog_deadzone);
analog_amplitude_y = (float)joy_ry / (float)(ANALOG_RANGE - analog_deadzone);
// Convert to polar coordinates: part 1
rs_radius = sqrt((double)(analog_amplitude_x * analog_amplitude_x) + (double)(analog_amplitude_y * analog_amplitude_y));
// Check if radius is above threshold
if (rs_radius > 0.5) {
// Convert to polar coordinates: part 2
rs_angle = atan2((double)analog_amplitude_y, (double)analog_amplitude_x);
// Adjust rotation offset...
rs_angle = (2.0 * PI) - (rs_angle + PI);
rs_angle = fmod(rs_angle - (0.125 * PI), 2.0 * PI);
if (rs_angle < 0)
rs_angle += 2.0 * PI;
// Convert angle into numpad key index
numpad_index = (unsigned)((rs_angle / (2.0 * PI)) * 8.0);
// Unnecessary safety check...
numpad_index = (numpad_index > 7) ? 7 : numpad_index;
// printf("numpad_index: %u\n", numpad_index);
if (numpad_index != _joypadnumpadLast) {
// Unset last key, if required
if (_joypadnumpadActive)
processKeyEvent(false, gampad_numpad_map[_joypadnumpadLast][0], (uint32_t)gampad_numpad_map[_joypadnumpadLast][1], 0);
// Set new key
processKeyEvent(true, gampad_numpad_map[numpad_index][0], (uint32_t)gampad_numpad_map[numpad_index][1], 0);
_joypadnumpadLast = numpad_index;
_joypadnumpadActive = true;
}
} else if (_joypadnumpadActive) {
processKeyEvent(false, gampad_numpad_map[_joypadnumpadLast][0], (uint32_t)gampad_numpad_map[_joypadnumpadLast][1], 0);
_joypadnumpadActive = false;
_joypadnumpadLast = 8;
}
} else if (_joypadnumpadActive) {
processKeyEvent(false, gampad_numpad_map[_joypadnumpadLast][0], (uint32_t)gampad_numpad_map[_joypadnumpadLast][1], 0);
_joypadnumpadActive = false;
_joypadnumpadLast = 8;
}
// Process input from physical mouse
// > X Axis
if (x != 0) {
status |= (STATUS_DOING_MOUSE | STATUS_DOING_X);
if (x > 0) {
// Reset accumulator when changing direction
_mouseXAcc = (_mouseXAcc < 0.0) ? 0.0 : _mouseXAcc;
}
if (x < 0) {
// Reset accumulator when changing direction
_mouseXAcc = (_mouseXAcc > 0.0) ? 0.0 : _mouseXAcc;
}
deltaAcc = (float)x * mouse_speed;
updateMouseXY(deltaAcc, &_mouseXAcc, 1);
}
// > Y Axis
if (y != 0) {
status |= (STATUS_DOING_MOUSE | STATUS_DOING_Y);
if (y > 0) {
// Reset accumulator when changing direction
_mouseYAcc = (_mouseYAcc < 0.0) ? 0.0 : _mouseYAcc;
}
if (y < 0) {
// Reset accumulator when changing direction
_mouseYAcc = (_mouseYAcc > 0.0) ? 0.0 : _mouseYAcc;
}
deltaAcc = (float)y * mouse_speed;
updateMouseXY(deltaAcc, &_mouseYAcc, 0);
}
if (status & STATUS_DOING_MOUSE) {
Common::Event ev;
ev.type = Common::EVENT_MOUSEMOVE;
ev.mouse.x = _mouseX;
ev.mouse.y = _mouseY;
ev.relMouse.x = status & STATUS_DOING_X ? _relMouseX : 0;
ev.relMouse.y = status & STATUS_DOING_Y ? _relMouseY : 0;
_events.push_back(ev);
}
for (int i = 0; i < 2; i++) {
Common::Event ev;
bool down = aCallback(0, RETRO_DEVICE_MOUSE, 0, retroButtons[i]);
if (down != _mouseButtons[i]) {
_mouseButtons[i] = down;
ev.type = eventID[i][down ? 0 : 1];
ev.mouse.x = _mouseX;
ev.mouse.y = _mouseY;
_events.push_back(ev);
}
}
}
void OSystem_libretro::processKeyEvent(bool down, unsigned keycode, uint32_t character, uint16_t key_modifiers) {
int _keyflags = 0;
_keyflags |= (key_modifiers & RETROKMOD_CTRL) ? Common::KBD_CTRL : 0;
_keyflags |= (key_modifiers & RETROKMOD_ALT) ? Common::KBD_ALT : 0;
_keyflags |= (key_modifiers & RETROKMOD_SHIFT) ? Common::KBD_SHIFT : 0;
_keyflags |= (key_modifiers & RETROKMOD_META) ? Common::KBD_META : 0;
_keyflags |= (key_modifiers & RETROKMOD_CAPSLOCK) ? Common::KBD_CAPS : 0;
_keyflags |= (key_modifiers & RETROKMOD_NUMLOCK) ? Common::KBD_NUM : 0;
_keyflags |= (key_modifiers & RETROKMOD_SCROLLOCK) ? Common::KBD_SCRL : 0;
Common::Event ev;
ev.type = down ? Common::EVENT_KEYDOWN : Common::EVENT_KEYUP;
ev.kbd.keycode = (Common::KeyCode)keycode;
ev.kbd.flags = _keyflags;
ev.kbd.ascii = keycode;
/* If shift was down then send upper case letter to engine */
if (ev.kbd.ascii >= 97 && ev.kbd.ascii <= 122 && (_keyflags & Common::KBD_SHIFT))
ev.kbd.ascii = ev.kbd.ascii & ~0x20;
_events.push_back(ev);
}

View File

@ -0,0 +1,157 @@
/* Copyright (C) 2023 Giovanni Cascione <ing.cascione@gmail.com>
*
* 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 3 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, see <http://www.gnu.org/licenses/>.
*
*/
#define FORBIDDEN_SYMBOL_EXCEPTION_time
#include <sys/time.h>
#if defined(__CELLOS_LV2__)
#include <sys/sys_time.h>
#elif (defined(GEKKO) && !defined(WIIU))
#include <ogc/lwp_watchdog.h>
#else
#include <time.h>
#endif
#include "common/tokenizer.h"
#include "common/config-manager.h"
#include "base/commandLine.h"
#include "backends/platform/libretro/include/libretro-os.h"
#include "backends/platform/libretro/include/libretro-defs.h"
void OSystem_libretro::getTimeAndDate(TimeDate &t, bool skipRecord) const {
time_t curTime = time(NULL);
#define YEAR0 1900
#define EPOCH_YR 1970
#define SECS_DAY (24L * 60L * 60L)
#define LEAPYEAR(year) (!((year) % 4) && (((year) % 100) || !((year) % 400)))
#define YEARSIZE(year) (LEAPYEAR(year) ? 366 : 365)
const int _ytab[2][12] = {{31, 28, 31, 30, 31, 30, 31, 31, 30, 31, 30, 31}, {31, 29, 31, 30, 31, 30, 31, 31, 30, 31, 30, 31}};
int year = EPOCH_YR;
unsigned long dayclock = (unsigned long)curTime % SECS_DAY;
unsigned long dayno = (unsigned long)curTime / SECS_DAY;
t.tm_sec = dayclock % 60;
t.tm_min = (dayclock % 3600) / 60;
t.tm_hour = dayclock / 3600;
t.tm_wday = (dayno + 4) % 7; /* day 0 was a thursday */
while (dayno >= YEARSIZE(year)) {
dayno -= YEARSIZE(year);
year++;
}
t.tm_year = year - YEAR0;
t.tm_mon = 0;
while (dayno >= _ytab[LEAPYEAR(year)][t.tm_mon]) {
dayno -= _ytab[LEAPYEAR(year)][t.tm_mon];
t.tm_mon++;
}
t.tm_mday = dayno + 1;
}
Common::String OSystem_libretro::getDefaultConfigFileName() {
return s_systemDir + "/scummvm.ini";
}
void OSystem_libretro::logMessage(LogMessageType::Type type, const char *message) {
retro_log_level loglevel = RETRO_LOG_INFO;
switch (type) {
case LogMessageType::kDebug:
loglevel = RETRO_LOG_DEBUG;
break;
case LogMessageType::kWarning:
loglevel = RETRO_LOG_WARN;
break;
case LogMessageType::kError:
loglevel = RETRO_LOG_ERROR;
break;
}
if (log_cb)
log_cb(loglevel, "%s\n", message);
}
bool OSystem_libretro::parseGameName(const Common::String &gameName, Common::String &engineId,
Common::String &gameId) {
Common::StringTokenizer tokenizer(gameName, ":");
Common::String token1, token2;
if (!tokenizer.empty()) {
token1 = tokenizer.nextToken();
}
if (!tokenizer.empty()) {
token2 = tokenizer.nextToken();
}
if (!tokenizer.empty()) {
return false; // Stray colon
}
if (!token1.empty() && !token2.empty()) {
engineId = token1;
gameId = token2;
return true;
} else if (!token1.empty()) {
engineId.clear();
gameId = token1;
return true;
}
return false;
}
int OSystem_libretro::testGame(const char *filedata, bool autodetect) {
Common::String game_id;
Common::String engine_id;
Common::String data = filedata;
int res = TEST_GAME_KO_NOT_FOUND;
PluginManager::instance().init();
PluginManager::instance().loadAllPlugins();
PluginManager::instance().loadDetectionPlugin();
if (autodetect) {
Common::FSNode dir(data);
Common::FSList files;
dir.getChildren(files, Common::FSNode::kListAll);
DetectionResults detectionResults = EngineMan.detectGames(files);
if (!detectionResults.listRecognizedGames().empty()) {
res = TEST_GAME_OK_ID_AUTODETECTED;
}
} else {
ConfMan.loadDefaultConfigFile(getDefaultConfigFileName().c_str());
if (ConfMan.hasGameDomain(data)) {
res = TEST_GAME_OK_TARGET_FOUND;
} else {
parseGameName(data, engine_id, game_id);
QualifiedGameList games = EngineMan.findGamesMatching(engine_id, game_id);
if (games.size() == 1) {
res = TEST_GAME_OK_ID_FOUND;
} else if (games.size() > 1) {
res = TEST_GAME_KO_MULTIPLE_RESULTS;
}
}
}
PluginManager::instance().unloadDetectionPlugin();
PluginManager::instance().unloadAllPlugins();
PluginManager::destroy();
return res;
}

File diff suppressed because it is too large Load Diff

View File

@ -15,9 +15,9 @@
*/
#include <stdio.h>
#include "backends/platform/libretro/include/libretro-threads.h"
#include "backends/platform/libretro/include/os.h"
#include <libretro.h>
#include "base/main.h"
#include "backends/platform/libretro/include/libretro-threads.h"
#define EMU_WAITING (1 << 0)
#define MAIN_WAITING (1 << 1)
@ -39,13 +39,8 @@ static scond_t *emu_cond;
static scond_t *main_cond;
#endif
static bool retro_current_thread_is_main() {
#ifdef USE_LIBCO
return (co_active() == main_thread);
#else
return (sthread_get_current_thread_id() == main_thread_id);
#endif
}
extern char cmd_params[20][200];
extern char cmd_params_num;
static void retro_exit_to_main_thread() {
#ifdef USE_LIBCO

View File

@ -1,4 +1,4 @@
/* Copyright (C) 2022 Giovanni Cascione <ing.cascione@gmail.com>
/* Copyright (C) 2023 Giovanni Cascione <ing.cascione@gmail.com>
*
* 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
@ -20,7 +20,7 @@
#include "common/timer.h"
#include "backends/platform/libretro/include/libretro-threads.h"
#include "backends/platform/libretro/include/libretro-timer.h"
#include "backends/platform/libretro/include/os.h"
#include "backends/platform/libretro/include/libretro-defs.h"
LibretroTimerManager::LibretroTimerManager(uint32 refresh_rate) {
if (! refresh_rate > 0)

View File

@ -52,7 +52,8 @@
#include "backends/platform/libretro/include/libretro-threads.h"
#include "backends/platform/libretro/include/libretro-core-options.h"
#include "backends/platform/libretro/include/os.h"
#include "backends/platform/libretro/include/libretro-os.h"
#include "backends/platform/libretro/include/libretro-defs.h"
static struct retro_game_info game_buf;
static struct retro_game_info * game_buf_ptr;
@ -377,7 +378,7 @@ static void exit_to_frontend(void) {
static void close_emu_thread(void) {
while (!retro_emu_thread_exited()) {
retroQuit();
LIBRETRO_G_SYSTEM->requestQuit();
retro_switch_to_emu_thread();
}
retro_deinit_emu_thread();
@ -583,14 +584,14 @@ void retro_init(void) {
log_cb(RETRO_LOG_INFO, "Frontend supports RGB565 -will use that instead of XRGB1555.\n");
#endif
retro_keyboard_callback cb = {retroKeyEvent};
retro_keyboard_callback cb = {LIBRETRO_G_SYSTEM->processKeyEvent};
environ_cb(RETRO_ENVIRONMENT_SET_KEYBOARD_CALLBACK, &cb);
g_system = retroBuildOS();
g_system = new OSystem_libretro();
}
void retro_deinit(void) {
retroDestroy();
LIBRETRO_G_SYSTEM->destroy();
free(sound_buffer);
}
@ -665,7 +666,7 @@ bool retro_load_game(const struct retro_game_info *game) {
return false;
}
test_game_status = retroTestGame(target_id, false);
test_game_status = LIBRETRO_G_SYSTEM->testGame(target_id, false);
} else {
if (detect_target.isDirectory()) {
parent_dir = detect_target;
@ -677,7 +678,7 @@ bool retro_load_game(const struct retro_game_info *game) {
}
}
test_game_status = retroTestGame(parent_dir.getPath().c_str(), true);
test_game_status = LIBRETRO_G_SYSTEM->testGame(parent_dir.getPath().c_str(), true);
}
// Preliminary game scan results
@ -770,7 +771,7 @@ void retro_run(void) {
/* No frame skipping if
- no incoming audio (e.g. GUI)
- doing a THREAD_SWITCH_UPDATE loop */
skip_frame = skip_frame && !(audio_status & AUDIO_STATUS_MUTE) && !(getThreadSwitchCaller() & THREAD_SWITCH_UPDATE);
skip_frame = skip_frame && !(audio_status & AUDIO_STATUS_MUTE) && !(LIBRETRO_G_SYSTEM->getThreadSwitchCaller() & THREAD_SWITCH_UPDATE);
/* Reset frameskip counter if not flagged */
if ((!skip_frame && frameskip_counter) || frameskip_counter >= FRAMESKIP_MAX) {
@ -831,7 +832,7 @@ void retro_run(void) {
/* Retrieve video */
if ((audio_video_enable & 1) && !skip_frame) {
const Graphics::Surface &screen = getScreen();
const Graphics::Surface &screen = LIBRETRO_G_SYSTEM->getScreen();
video_cb(screen.getPixels(), screen.w, screen.h, screen.pitch);
}
@ -850,10 +851,10 @@ void retro_run(void) {
current_frame++;
} while (getThreadSwitchCaller() & THREAD_SWITCH_UPDATE);
} while (LIBRETRO_G_SYSTEM->getThreadSwitchCaller() & THREAD_SWITCH_UPDATE);
poll_cb();
retroProcessMouse(input_cb, retro_device, gampad_cursor_speed, gamepad_acceleration_time, analog_response_is_quadratic, analog_deadzone, mouse_speed);
LIBRETRO_G_SYSTEM->processMouse(input_cb, retro_device, gampad_cursor_speed, gamepad_acceleration_time, analog_response_is_quadratic, analog_deadzone, mouse_speed);
}
}
@ -865,7 +866,7 @@ void retro_reset(void) {
close_emu_thread();
init_command_params();
retro_load_game(game_buf_ptr);
retroReset();
LIBRETRO_G_SYSTEM->resetQuit();
}
// Stubs