RetroArch/frontend/frontend_driver.c

574 lines
14 KiB
C
Raw Normal View History

/* RetroArch - A frontend for libretro.
2014-01-01 00:50:59 +00:00
* Copyright (C) 2010-2014 - Hans-Kristian Arntzen
2017-01-22 12:40:32 +00:00
* Copyright (C) 2011-2017 - 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 <http://www.gnu.org/licenses/>.
*/
#include <stdio.h>
#include <string.h>
2016-02-03 14:23:13 +00:00
#include <compat/strl.h>
2016-09-16 15:25:47 +00:00
#include <string/stdstring.h>
#include <retro_miscellaneous.h>
#include <libretro.h>
2016-02-03 14:23:13 +00:00
#if defined(_3DS)
#include <3ds.h>
#endif
#ifdef HAVE_CONFIG_H
#include "../config.h"
#endif
#include "frontend_driver.h"
2019-01-03 12:55:43 +00:00
#ifndef __WINRT__
#if defined(WINAPI_FAMILY) && WINAPI_FAMILY == WINAPI_FAMILY_PHONE_APP
#define __WINRT__
#endif
#endif
2020-01-06 14:35:35 +00:00
static frontend_ctx_driver_t frontend_ctx_null = {
NULL, /* environment_get */
NULL, /* init */
NULL, /* deinit */
NULL, /* exitspawn */
NULL, /* process_args */
NULL, /* exec */
NULL, /* set_fork */
NULL, /* shutdown */
NULL, /* get_name */
NULL, /* get_os */
NULL, /* get_rating */
NULL, /* load_content */
NULL, /* get_architecture */
NULL, /* get_powerstate */
NULL, /* parse_drive_list */
NULL, /* get_mem_total */
NULL, /* get_mem_free */
NULL, /* install_signal_handler */
NULL, /* get_sighandler_state */
NULL, /* set_sighandler_state */
NULL, /* destroy_sighandler_state */
NULL, /* attach_console */
NULL, /* detach_console */
#ifdef HAVE_LAKKA
NULL, /* get_lakka_version */
#endif
NULL, /* watch_path_for_changes */
NULL, /* check_for_path_changes */
NULL, /* set_sustained_performance_mode */
NULL, /* get_cpu_model_name */
NULL, /* get_user_language */
NULL, /* is_narrator_running */
NULL, /* accessibility_speak */
2020-01-06 14:35:35 +00:00
"null",
NULL, /* get_video_driver */
2020-01-06 14:35:35 +00:00
};
static frontend_ctx_driver_t *frontend_ctx_drivers[] = {
#if defined(EMSCRIPTEN)
&frontend_ctx_emscripten,
#elif defined(__CELLOS_LV2__)
&frontend_ctx_ps3,
#endif
#if defined(_XBOX)
&frontend_ctx_xdk,
#endif
#if defined(GEKKO)
&frontend_ctx_gx,
#endif
#if defined(WIIU)
2016-10-27 22:02:40 +00:00
&frontend_ctx_wiiu,
#endif
2013-07-27 15:16:46 +00:00
#if defined(__QNX__)
&frontend_ctx_qnx,
#endif
#if defined(__APPLE__) && defined(__MACH__)
&frontend_ctx_darwin,
2013-11-03 15:38:56 +00:00
#endif
#if defined(__linux__) || (defined(BSD) && !defined(__MACH__))
&frontend_ctx_unix,
2016-07-08 00:13:19 +00:00
#endif
#if defined(PSP) || defined(VITA)
&frontend_ctx_psp,
2015-04-01 21:14:13 +00:00
#endif
2018-09-18 06:08:06 +00:00
#if defined(PS2)
2018-10-17 17:02:50 +00:00
&frontend_ctx_ps2,
2018-09-18 06:08:06 +00:00
#endif
2015-04-01 21:14:13 +00:00
#if defined(_3DS)
&frontend_ctx_ctr,
2015-04-07 19:51:57 +00:00
#endif
#if defined(SWITCH) && defined(HAVE_LIBNX)
&frontend_ctx_switch,
#endif
2019-01-03 12:55:43 +00:00
#if defined(_WIN32) && !defined(_XBOX) && !defined(__WINRT__)
2015-04-07 20:12:28 +00:00
&frontend_ctx_win32,
#endif
2019-01-03 12:55:43 +00:00
#if defined(__WINRT__)
&frontend_ctx_uwp,
#endif
#ifdef XENON
&frontend_ctx_xenon,
2017-01-24 05:55:55 +00:00
#endif
#ifdef DJGPP
&frontend_ctx_dos,
2018-09-15 06:50:08 +00:00
#endif
#ifdef SWITCH
&frontend_ctx_switch,
2018-12-31 13:16:27 +00:00
#endif
#if defined(ORBIS)
&frontend_ctx_orbis,
2013-07-27 15:40:21 +00:00
#endif
&frontend_ctx_null,
NULL
};
2016-01-02 00:38:02 +00:00
#ifndef IS_SALAMANDER
2020-05-29 04:31:15 +00:00
/* TODO/FIXME - static public global variable */
static frontend_ctx_driver_t *current_frontend_ctx;
2016-01-02 00:38:02 +00:00
#endif
/**
* frontend_ctx_find_driver:
* @ident : Identifier name of driver to find.
*
* Finds driver with @ident. Does not initialize.
*
* Returns: pointer to driver if successful, otherwise NULL.
**/
frontend_ctx_driver_t *frontend_ctx_find_driver(const char *ident)
{
unsigned i;
2015-01-11 05:57:35 +00:00
for (i = 0; frontend_ctx_drivers[i]; i++)
{
2016-09-16 15:25:47 +00:00
if (string_is_equal(frontend_ctx_drivers[i]->ident, ident))
return frontend_ctx_drivers[i];
}
return NULL;
}
/**
* frontend_ctx_init_first:
*
* Finds first suitable driver and initialize.
*
* Returns: pointer to first suitable driver, otherwise NULL.
**/
frontend_ctx_driver_t *frontend_ctx_init_first(void)
{
return frontend_ctx_drivers[0];
}
2015-04-07 22:08:53 +00:00
2016-02-03 13:36:34 +00:00
bool frontend_driver_get_core_extension(char *s, size_t len)
{
#ifdef HAVE_DYNAMIC
#ifdef _WIN32
strlcpy(s, "dll", len);
return true;
#elif defined(__APPLE__) || defined(__MACH__)
strlcpy(s, "dylib", len);
return true;
#else
strlcpy(s, "so", len);
return true;
#endif
#else
#if defined(__CELLOS_LV2__)
strlcpy(s, "self|bin", len);
return true;
#elif defined(PSP)
strlcpy(s, "pbp", len);
return true;
#elif defined(VITA)
2016-09-07 23:25:14 +00:00
strlcpy(s, "self|bin", len);
2016-02-03 13:36:34 +00:00
return true;
2018-09-18 06:08:06 +00:00
#elif defined(PS2)
strlcpy(s, "elf", len);
return true;
2016-02-03 13:36:34 +00:00
#elif defined(_XBOX1)
strlcpy(s, "xbe", len);
return true;
#elif defined(_XBOX360)
strlcpy(s, "xex", len);
return true;
#elif defined(GEKKO)
strlcpy(s, "dol", len);
return true;
#elif defined(HW_WUP)
strlcpy(s, "rpx|elf", len);
return true;
#elif defined(__linux__)
strlcpy(s, "elf", len);
return true;
2018-09-15 00:29:40 +00:00
#elif defined(HAVE_LIBNX)
strlcpy(s, "nro", len);
return true;
2020-05-12 15:39:04 +00:00
#elif defined(DJGPP)
strlcpy(s, "exe", len);
return true;
#elif defined(_3DS)
if (envIsHomebrew())
strlcpy(s, "3dsx", len);
else
2018-05-16 17:41:25 +00:00
strlcpy(s, "cia", len);
return true;
2016-02-03 13:36:34 +00:00
#else
return false;
#endif
#endif
}
bool frontend_driver_get_salamander_basename(char *s, size_t len)
{
#ifdef HAVE_DYNAMIC
return false;
#else
#if defined(__CELLOS_LV2__)
strlcpy(s, "EBOOT.BIN", len);
return true;
#elif defined(PSP)
strlcpy(s, "EBOOT.PBP", len);
return true;
#elif defined(VITA)
2016-09-07 23:25:14 +00:00
strlcpy(s, "eboot.bin", len);
return true;
2018-09-18 06:08:06 +00:00
#elif defined(PS2)
strlcpy(s, "eboot.elf", len);
return true;
#elif defined(_XBOX1)
strlcpy(s, "default.xbe", len);
return true;
#elif defined(_XBOX360)
strlcpy(s, "default.xex", len);
return true;
#elif defined(HW_RVL)
strlcpy(s, "boot.dol", len);
return true;
#elif defined(HW_WUP)
strlcpy(s, "retroarch.rpx", len);
return true;
#elif defined(_3DS)
strlcpy(s, "retroarch.core", len);
return true;
2020-05-12 15:39:04 +00:00
#elif defined(DJGPP)
strlcpy(s, "retrodos.exe", len);
return true;
2018-09-15 00:29:40 +00:00
#elif defined(SWITCH)
strlcpy(s, "retroarch_switch.nro", len);
return true;
#else
return false;
#endif
#endif
}
2015-04-10 05:46:54 +00:00
#ifndef IS_SALAMANDER
frontend_ctx_driver_t *frontend_get_ptr(void)
2015-04-07 22:08:53 +00:00
{
return current_frontend_ctx;
2015-04-07 22:08:53 +00:00
}
2015-06-15 20:45:02 +00:00
int frontend_driver_parse_drive_list(void *data, bool load_content)
2015-06-15 20:45:02 +00:00
{
frontend_ctx_driver_t *frontend = frontend_get_ptr();
2015-06-15 20:45:02 +00:00
if (!frontend || !frontend->parse_drive_list)
return -1;
return frontend->parse_drive_list(data, load_content);
2015-06-15 20:45:02 +00:00
}
2015-06-15 20:47:20 +00:00
2015-12-05 06:53:02 +00:00
void frontend_driver_content_loaded(void)
{
frontend_ctx_driver_t *frontend = frontend_get_ptr();
if (!frontend || !frontend->content_loaded)
return;
frontend->content_loaded();
}
2015-12-05 06:57:45 +00:00
2016-02-03 10:56:01 +00:00
bool frontend_driver_has_fork(void)
{
frontend_ctx_driver_t *frontend = frontend_get_ptr();
if (!frontend || !frontend->set_fork)
return false;
return true;
}
2016-02-04 16:46:50 +00:00
bool frontend_driver_set_fork(enum frontend_fork fork_mode)
2015-12-05 06:57:45 +00:00
{
frontend_ctx_driver_t *frontend = frontend_get_ptr();
2016-02-04 10:44:10 +00:00
if (!frontend_driver_has_fork())
return false;
2016-02-04 16:46:50 +00:00
return frontend->set_fork(fork_mode);
2015-12-05 06:57:45 +00:00
}
2015-12-05 07:10:01 +00:00
void frontend_driver_process_args(int *argc, char *argv[])
{
frontend_ctx_driver_t *frontend = frontend_get_ptr();
if (!frontend || !frontend->process_args)
return;
frontend->process_args(argc, argv);
}
bool frontend_driver_is_inited(void)
{
frontend_ctx_driver_t *frontend = frontend_get_ptr();
if (!frontend)
return false;
return true;
}
void frontend_driver_init_first(void *args)
{
current_frontend_ctx = (frontend_ctx_driver_t*)frontend_ctx_init_first();
if (current_frontend_ctx && current_frontend_ctx->init)
current_frontend_ctx->init(args);
}
void frontend_driver_free(void)
{
current_frontend_ctx = NULL;
}
environment_get_t frontend_driver_environment_get_ptr(void)
{
frontend_ctx_driver_t *frontend = frontend_get_ptr();
if (!frontend)
return NULL;
return frontend->environment_get;
}
bool frontend_driver_has_get_video_driver_func(void)
{
frontend_ctx_driver_t *frontend = frontend_get_ptr();
if (!frontend || !frontend->get_video_driver)
return false;
return true;
}
const struct video_driver *frontend_driver_get_video_driver(void)
{
frontend_ctx_driver_t *frontend = frontend_get_ptr();
if (!frontend || !frontend->get_video_driver)
return NULL;
return frontend->get_video_driver();
}
void frontend_driver_exitspawn(char *s, size_t len, char *args)
{
frontend_ctx_driver_t *frontend = frontend_get_ptr();
if (!frontend || !frontend->exitspawn)
return;
frontend->exitspawn(s, len, args);
}
void frontend_driver_deinit(void *args)
{
frontend_ctx_driver_t *frontend = frontend_get_ptr();
if (!frontend || !frontend->deinit)
return;
frontend->deinit(args);
}
void frontend_driver_shutdown(bool a)
{
frontend_ctx_driver_t *frontend = frontend_get_ptr();
if (!frontend || !frontend->shutdown)
2015-12-05 08:13:14 +00:00
return;
frontend->shutdown(a);
}
enum frontend_architecture frontend_driver_get_cpu_architecture(void)
{
frontend_ctx_driver_t *frontend = frontend_get_ptr();
if (!frontend || !frontend->get_architecture)
return FRONTEND_ARCH_NONE;
return frontend->get_architecture();
}
2016-06-12 00:33:07 +00:00
const void *frontend_driver_get_cpu_architecture_str(
char *architecture, size_t size)
{
const frontend_ctx_driver_t
*frontend = frontend_get_ptr();
enum frontend_architecture arch = frontend_driver_get_cpu_architecture();
switch (arch)
{
case FRONTEND_ARCH_X86:
strlcpy(architecture, "x86", size);
break;
case FRONTEND_ARCH_X86_64:
strlcpy(architecture, "x64", size);
break;
case FRONTEND_ARCH_PPC:
strlcpy(architecture, "PPC", size);
break;
case FRONTEND_ARCH_ARM:
strlcpy(architecture, "ARM", size);
break;
case FRONTEND_ARCH_ARMV7:
strlcpy(architecture, "ARMv7", size);
break;
case FRONTEND_ARCH_ARMV8:
strlcpy(architecture, "ARMv8", size);
break;
case FRONTEND_ARCH_MIPS:
strlcpy(architecture, "MIPS", size);
break;
case FRONTEND_ARCH_TILE:
strlcpy(architecture, "Tilera", size);
break;
case FRONTEND_ARCH_NONE:
default:
strlcpy(architecture, "N/A", size);
break;
}
return frontend;
}
uint64_t frontend_driver_get_total_memory(void)
2016-06-12 00:33:07 +00:00
{
frontend_ctx_driver_t *frontend = frontend_get_ptr();
if (!frontend || !frontend->get_total_mem)
return 0;
return frontend->get_total_mem();
}
2016-06-12 10:45:13 +00:00
uint64_t frontend_driver_get_free_memory(void)
2016-06-12 10:45:13 +00:00
{
frontend_ctx_driver_t *frontend = frontend_get_ptr();
if (!frontend || !frontend->get_free_mem)
2016-06-12 10:45:13 +00:00
return 0;
return frontend->get_free_mem();
2016-06-12 10:45:13 +00:00
}
2016-07-08 00:10:40 +00:00
2016-07-08 00:24:56 +00:00
void frontend_driver_install_signal_handler(void)
2016-07-08 00:10:40 +00:00
{
frontend_ctx_driver_t *frontend = frontend_get_ptr();
if (!frontend || !frontend->install_signal_handler)
return;
frontend->install_signal_handler();
}
2016-07-08 00:20:42 +00:00
2016-07-08 00:24:56 +00:00
int frontend_driver_get_signal_handler_state(void)
2016-07-08 00:20:42 +00:00
{
frontend_ctx_driver_t *frontend = frontend_get_ptr();
if (!frontend || !frontend->get_signal_handler_state)
return -1;
return frontend->get_signal_handler_state();
}
2016-07-08 00:24:56 +00:00
2016-07-08 10:48:01 +00:00
void frontend_driver_set_signal_handler_state(int value)
{
frontend_ctx_driver_t *frontend = frontend_get_ptr();
if (!frontend || !frontend->set_signal_handler_state)
return;
frontend->set_signal_handler_state(value);
}
void frontend_driver_attach_console(void)
{
frontend_ctx_driver_t *frontend = frontend_get_ptr();
if (!frontend || !frontend->attach_console)
return;
frontend->attach_console();
}
void frontend_driver_detach_console(void)
{
frontend_ctx_driver_t *frontend = frontend_get_ptr();
if (!frontend || !frontend->detach_console)
return;
frontend->detach_console();
}
2016-07-08 00:24:56 +00:00
void frontend_driver_destroy_signal_handler_state(void)
{
frontend_ctx_driver_t *frontend = frontend_get_ptr();
if (!frontend || !frontend->destroy_signal_handler_state)
return;
frontend->destroy_signal_handler_state();
}
bool frontend_driver_can_watch_for_changes(void)
{
frontend_ctx_driver_t *frontend = frontend_get_ptr();
if (!frontend || !frontend->watch_path_for_changes)
return false;
return true;
}
void frontend_driver_watch_path_for_changes(struct string_list *list, int flags, path_change_data_t **change_data)
{
frontend_ctx_driver_t *frontend = frontend_get_ptr();
if (!frontend || !frontend->watch_path_for_changes)
return;
frontend->watch_path_for_changes(list, flags, change_data);
}
bool frontend_driver_check_for_path_changes(path_change_data_t *change_data)
{
frontend_ctx_driver_t *frontend = frontend_get_ptr();
if (!frontend || !frontend->check_for_path_changes)
return false;
return frontend->check_for_path_changes(change_data);
}
void frontend_driver_set_sustained_performance_mode(bool on)
{
frontend_ctx_driver_t *frontend = frontend_get_ptr();
if (!frontend || !frontend->set_sustained_performance_mode)
return;
frontend->set_sustained_performance_mode(on);
}
const char* frontend_driver_get_cpu_model_name(void)
{
frontend_ctx_driver_t *frontend = frontend_get_ptr();
if (!frontend || !frontend->get_cpu_model_name)
return NULL;
return frontend->get_cpu_model_name();
}
enum retro_language frontend_driver_get_user_language(void)
{
frontend_ctx_driver_t *frontend = frontend_get_ptr();
if (!frontend || !frontend->get_user_language)
return RETRO_LANGUAGE_ENGLISH;
return frontend->get_user_language();
}
bool frontend_driver_is_narrator_running(void)
{
frontend_ctx_driver_t *frontend = frontend_get_ptr();
if (!frontend || !frontend->is_narrator_running)
return false;
return frontend->is_narrator_running();
}
2015-12-05 06:53:02 +00:00
#endif