RetroArch/dynamic.c

1273 lines
39 KiB
C
Raw Normal View History

2014-02-07 07:14:02 +00:00
/* RetroArch - A frontend for libretro.
2014-01-01 00:50:59 +00:00
* Copyright (C) 2010-2014 - Hans-Kristian Arntzen
2016-01-10 03:06:50 +00:00
* Copyright (C) 2011-2016 - Daniel De Matteis
*
2012-04-21 21:13:50 +00:00
* RetroArch is free software: you can redistribute it and/or modify it under the terms
2010-12-30 12:54:49 +00:00
* 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.
*
2012-04-21 21:13:50 +00:00
* RetroArch is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY;
2010-12-30 12:54:49 +00:00
* without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR
* PURPOSE. See the GNU General Public License for more details.
*
2012-04-21 21:31:57 +00:00
* You should have received a copy of the GNU General Public License along with RetroArch.
2010-12-30 12:54:49 +00:00
* If not, see <http://www.gnu.org/licenses/>.
*/
#include <string.h>
2013-03-22 19:56:23 +00:00
#include <ctype.h>
2011-01-07 16:59:53 +00:00
#include <file/file_path.h>
#include <compat/strl.h>
#include <compat/posix_string.h>
2015-12-05 19:26:37 +00:00
#include <dynamic/dylib.h>
#include <string/stdstring.h>
2015-09-15 03:22:57 +00:00
#include <boolean.h>
2011-01-07 16:59:53 +00:00
#ifdef HAVE_CONFIG_H
2010-12-30 12:54:49 +00:00
#include "config.h"
2011-01-07 16:59:53 +00:00
#endif
#include "command_event.h"
2015-11-23 18:40:09 +00:00
#include "audio/audio_driver.h"
#include "camera/camera_driver.h"
#include "location/location_driver.h"
#include "record/record_driver.h"
2015-11-30 02:30:04 +00:00
#include "performance.h"
#include "system.h"
#include "libretro_private.h"
#include "cores/internal_cores.h"
2015-12-24 20:17:44 +00:00
#include "frontend/frontend_driver.h"
2015-01-09 16:40:47 +00:00
#include "retroarch.h"
2015-06-02 15:17:46 +00:00
#include "configuration.h"
#include "general.h"
2015-07-02 12:04:15 +00:00
#include "msg_hash.h"
2015-11-23 11:03:38 +00:00
#include "verbosity.h"
2010-12-30 12:54:49 +00:00
2012-04-05 09:47:43 +00:00
#ifdef HAVE_DYNAMIC
2015-09-15 03:22:57 +00:00
#define SYMBOL(x) do { \
2012-04-07 09:55:37 +00:00
function_t func = dylib_proc(lib_handle, #x); \
memcpy(&core.x, &func, sizeof(func)); \
if (core.x == NULL) { RARCH_ERR("Failed to load symbol: \"%s\"\n", #x); retro_fail(1, "init_libretro_sym()"); } \
2011-10-15 10:56:48 +00:00
} while (0)
2010-12-30 12:54:49 +00:00
2013-04-14 14:24:19 +00:00
static dylib_t lib_handle;
2012-04-05 09:47:43 +00:00
#else
#define SYMBOL(x) core.x = x
2011-01-19 12:25:18 +00:00
#endif
2010-12-30 12:54:49 +00:00
#define SYMBOL_DUMMY(x) core.x = libretro_dummy_##x
2013-04-14 14:24:19 +00:00
#ifdef HAVE_FFMPEG
#define SYMBOL_FFMPEG(x) core.x = libretro_ffmpeg_##x
#endif
2015-06-28 16:55:00 +00:00
#ifdef HAVE_IMAGEVIEWER
#define SYMBOL_IMAGEVIEWER(x) core.x = libretro_imageviewer_##x
2015-06-28 16:55:00 +00:00
#endif
2015-06-28 15:02:01 +00:00
struct retro_core_t core;
static bool ignore_environment_cb;
2014-09-23 01:07:33 +00:00
2016-01-25 06:02:26 +00:00
const struct retro_subsystem_info *libretro_find_subsystem_info(
const struct retro_subsystem_info *info, unsigned num_info,
const char *ident)
{
unsigned i;
for (i = 0; i < num_info; i++)
{
if (string_is_equal(info[i].ident, ident))
return &info[i];
else if (string_is_equal(info[i].desc, ident))
return &info[i];
}
return NULL;
}
/**
* libretro_find_controller_description:
* @info : Pointer to controller info handle.
* @id : Identifier of controller to search
* for.
*
* Search for a controller of type @id in @info.
*
* Returns: controller description of found controller on success,
* otherwise NULL.
**/
const struct retro_controller_description *
libretro_find_controller_description(
const struct retro_controller_info *info, unsigned id)
{
unsigned i;
for (i = 0; i < info->num_types; i++)
{
if (info->types[i].id != id)
continue;
return &info->types[i];
}
return NULL;
}
2012-10-14 18:09:26 +00:00
#ifdef HAVE_DYNAMIC
2014-07-28 17:37:25 +00:00
static bool *load_no_content_hook;
2014-09-02 03:57:53 +00:00
static bool environ_cb_get_system_info(unsigned cmd, void *data)
{
switch (cmd)
{
case RETRO_ENVIRONMENT_SET_SUPPORT_NO_GAME:
2014-07-28 17:37:25 +00:00
*load_no_content_hook = *(const bool*)data;
break;
default:
return false;
}
return true;
}
2015-01-09 20:30:07 +00:00
/**
* libretro_get_environment_info:
* @func : Function pointer for get_environment_info.
* @load_no_content : If true, core should be able to auto-start
* without any content loaded.
*
* Sets environment callback in order to get statically known
2015-01-09 20:30:07 +00:00
* information from it.
*
* Fetched via environment callbacks instead of
* retro_get_system_info(), as this info is part of extensions.
*
* Should only be called once right after core load to
2015-01-09 20:30:07 +00:00
* avoid overwriting the "real" environ callback.
*
* For statically linked cores, pass retro_set_environment as argument.
*/
2014-09-02 03:57:53 +00:00
void libretro_get_environment_info(void (*func)(retro_environment_t),
bool *load_no_content)
{
2014-07-28 17:37:25 +00:00
load_no_content_hook = load_no_content;
2014-09-09 03:24:32 +00:00
/* load_no_content gets set in this callback. */
func(environ_cb_get_system_info);
2014-09-21 12:33:51 +00:00
2016-01-26 04:03:41 +00:00
/* It's possible that we just set get_system_info callback
* to the currently running core.
*
2014-09-21 12:33:51 +00:00
* Make sure we reset it to the actual environment callback.
2016-01-26 04:03:41 +00:00
* Ignore any environment callbacks here in case we're running
* on the non-current core. */
ignore_environment_cb = true;
2014-09-21 12:33:51 +00:00
func(rarch_environment_cb);
ignore_environment_cb = false;
}
2014-09-02 03:57:53 +00:00
static dylib_t libretro_get_system_info_lib(const char *path,
struct retro_system_info *info, bool *load_no_content)
2013-03-17 19:08:24 +00:00
{
dylib_t lib = dylib_load(path);
2015-06-26 15:46:13 +00:00
void (*proc)(struct retro_system_info*);
2013-03-17 19:08:24 +00:00
if (!lib)
{
RARCH_ERR("Failed to open libretro core: \"%s\"\n",
path);
RARCH_ERR("Error(s): %s\n", dylib_error());
2013-03-17 19:08:24 +00:00
return NULL;
}
2013-03-17 19:08:24 +00:00
2015-06-26 15:46:13 +00:00
proc = (void (*)(struct retro_system_info*))
dylib_proc(lib, "retro_get_system_info");
2013-03-17 19:08:24 +00:00
if (!proc)
{
dylib_close(lib);
return NULL;
}
proc(info);
2014-07-28 17:37:25 +00:00
if (load_no_content)
{
2015-06-26 15:46:13 +00:00
void (*set_environ)(retro_environment_t);
2014-07-28 17:37:25 +00:00
*load_no_content = false;
2015-06-26 15:46:13 +00:00
set_environ = (void (*)(retro_environment_t))
dylib_proc(lib, "retro_set_environment");
if (!set_environ)
return lib;
2014-07-28 17:37:25 +00:00
libretro_get_environment_info(set_environ, load_no_content);
}
2013-03-17 19:08:24 +00:00
return lib;
}
2015-01-09 20:30:07 +00:00
/**
* libretro_get_system_info:
* @path : Path to libretro library.
* @info : Pointer to system info information.
* @load_no_content : If true, core should be able to auto-start
* without any content loaded.
*
* Gets system info from an arbitrary lib.
* The struct returned must be freed as strings are allocated dynamically.
*
* Returns: true (1) if successful, otherwise false (0).
**/
2014-09-02 03:57:53 +00:00
bool libretro_get_system_info(const char *path,
struct retro_system_info *info, bool *load_no_content)
2013-03-17 19:08:24 +00:00
{
struct retro_system_info dummy_info = {0};
2014-09-02 03:57:53 +00:00
dylib_t lib = libretro_get_system_info_lib(path,
&dummy_info, load_no_content);
2013-03-17 19:08:24 +00:00
if (!lib)
return false;
memcpy(info, &dummy_info, sizeof(*info));
info->library_name = strdup(dummy_info.library_name);
info->library_version = strdup(dummy_info.library_version);
if (dummy_info.valid_extensions)
info->valid_extensions = strdup(dummy_info.valid_extensions);
2013-03-17 19:08:24 +00:00
dylib_close(lib);
return true;
}
2015-01-09 20:30:07 +00:00
/**
* libretro_free_system_info:
* @info : Pointer to system info information.
*
* Frees system information.
**/
2013-03-17 19:08:24 +00:00
void libretro_free_system_info(struct retro_system_info *info)
{
if (!info)
return;
2013-03-17 19:08:24 +00:00
free((void*)info->library_name);
free((void*)info->library_version);
free((void*)info->valid_extensions);
memset(info, 0, sizeof(*info));
}
static void load_dynamic_core(void)
{
settings_t *settings = config_get_ptr();
function_t sym = dylib_proc(NULL, "retro_init");
if (sym)
{
/* Try to verify that -lretro was not linked in from other modules
* since loading it dynamically and with -l will fail hard. */
2016-01-26 04:07:37 +00:00
RARCH_ERR("Serious problem. RetroArch wants to load libretro cores"
"dyamically, but it is already linked.\n");
RARCH_ERR("This could happen if other modules RetroArch depends on "
"link against libretro directly.\n");
RARCH_ERR("Proceeding could cause a crash. Aborting ...\n");
retro_fail(1, "init_libretro_sym()");
}
if (!*settings->libretro)
{
2016-01-26 04:07:37 +00:00
RARCH_ERR("RetroArch is built for dynamic libretro cores, but "
"libretro_path is not set. Cannot continue.\n");
retro_fail(1, "init_libretro_sym()");
}
/* Need to use absolute path for this setting. It can be
* saved to content history, and a relative path would
* break in that scenario. */
path_resolve_realpath(settings->libretro, sizeof(settings->libretro));
RARCH_LOG("Loading dynamic libretro core from: \"%s\"\n",
settings->libretro);
lib_handle = dylib_load(settings->libretro);
if (!lib_handle)
{
RARCH_ERR("Failed to open libretro core: \"%s\"\n",
settings->libretro);
RARCH_ERR("Error(s): %s\n", dylib_error());
retro_fail(1, "load_dynamic()");
}
}
#endif
2015-01-15 01:48:40 +00:00
/**
* load_symbols:
2015-06-20 21:42:30 +00:00
* @type : Type of core to be loaded.
* If CORE_TYPE_DUMMY, will
2015-06-20 21:42:30 +00:00
* load dummy symbols.
2015-01-15 01:48:40 +00:00
*
* Setup libretro callback symbols.
**/
2015-06-20 21:42:30 +00:00
static void load_symbols(enum rarch_core_type type)
2010-12-30 12:54:49 +00:00
{
2015-06-20 21:42:30 +00:00
switch (type)
{
2015-06-20 21:42:30 +00:00
case CORE_TYPE_PLAIN:
#ifdef HAVE_DYNAMIC
load_dynamic_core();
2015-06-20 21:42:30 +00:00
#endif
2015-09-15 03:22:57 +00:00
SYMBOL(retro_init);
SYMBOL(retro_deinit);
2010-12-30 12:54:49 +00:00
2015-09-15 03:22:57 +00:00
SYMBOL(retro_api_version);
SYMBOL(retro_get_system_info);
SYMBOL(retro_get_system_av_info);
2010-12-30 12:54:49 +00:00
2015-09-15 03:22:57 +00:00
SYMBOL(retro_set_environment);
SYMBOL(retro_set_video_refresh);
SYMBOL(retro_set_audio_sample);
SYMBOL(retro_set_audio_sample_batch);
SYMBOL(retro_set_input_poll);
SYMBOL(retro_set_input_state);
2012-04-05 09:47:43 +00:00
2015-09-15 03:22:57 +00:00
SYMBOL(retro_set_controller_port_device);
2012-04-05 09:47:43 +00:00
2015-09-15 03:22:57 +00:00
SYMBOL(retro_reset);
SYMBOL(retro_run);
2012-04-05 09:47:43 +00:00
2015-09-15 03:22:57 +00:00
SYMBOL(retro_serialize_size);
SYMBOL(retro_serialize);
SYMBOL(retro_unserialize);
2012-04-05 09:47:43 +00:00
2015-09-15 03:22:57 +00:00
SYMBOL(retro_cheat_reset);
SYMBOL(retro_cheat_set);
2012-04-05 09:47:43 +00:00
2015-09-15 03:22:57 +00:00
SYMBOL(retro_load_game);
SYMBOL(retro_load_game_special);
2010-12-30 12:54:49 +00:00
2015-09-15 03:22:57 +00:00
SYMBOL(retro_unload_game);
SYMBOL(retro_get_region);
SYMBOL(retro_get_memory_data);
SYMBOL(retro_get_memory_size);
2015-06-20 21:42:30 +00:00
break;
case CORE_TYPE_DUMMY:
2015-09-15 03:22:57 +00:00
SYMBOL_DUMMY(retro_init);
SYMBOL_DUMMY(retro_deinit);
2013-04-14 14:24:19 +00:00
2015-09-15 03:22:57 +00:00
SYMBOL_DUMMY(retro_api_version);
SYMBOL_DUMMY(retro_get_system_info);
SYMBOL_DUMMY(retro_get_system_av_info);
2013-04-14 14:24:19 +00:00
2015-09-15 03:22:57 +00:00
SYMBOL_DUMMY(retro_set_environment);
SYMBOL_DUMMY(retro_set_video_refresh);
SYMBOL_DUMMY(retro_set_audio_sample);
SYMBOL_DUMMY(retro_set_audio_sample_batch);
SYMBOL_DUMMY(retro_set_input_poll);
SYMBOL_DUMMY(retro_set_input_state);
2013-04-14 14:24:19 +00:00
2015-09-15 03:22:57 +00:00
SYMBOL_DUMMY(retro_set_controller_port_device);
2013-04-14 14:24:19 +00:00
2015-09-15 03:22:57 +00:00
SYMBOL_DUMMY(retro_reset);
SYMBOL_DUMMY(retro_run);
2013-04-14 14:24:19 +00:00
2015-09-15 03:22:57 +00:00
SYMBOL_DUMMY(retro_serialize_size);
SYMBOL_DUMMY(retro_serialize);
SYMBOL_DUMMY(retro_unserialize);
2013-04-14 14:24:19 +00:00
2015-09-15 03:22:57 +00:00
SYMBOL_DUMMY(retro_cheat_reset);
SYMBOL_DUMMY(retro_cheat_set);
2013-04-14 14:24:19 +00:00
2015-09-15 03:22:57 +00:00
SYMBOL_DUMMY(retro_load_game);
SYMBOL_DUMMY(retro_load_game_special);
2013-04-14 14:24:19 +00:00
2015-09-15 03:22:57 +00:00
SYMBOL_DUMMY(retro_unload_game);
SYMBOL_DUMMY(retro_get_region);
SYMBOL_DUMMY(retro_get_memory_data);
SYMBOL_DUMMY(retro_get_memory_size);
2015-06-20 21:42:30 +00:00
break;
case CORE_TYPE_FFMPEG:
2016-01-25 06:04:54 +00:00
#ifdef HAVE_FFMPEG
2015-09-15 03:22:57 +00:00
SYMBOL_FFMPEG(retro_init);
SYMBOL_FFMPEG(retro_deinit);
2015-09-15 03:22:57 +00:00
SYMBOL_FFMPEG(retro_api_version);
SYMBOL_FFMPEG(retro_get_system_info);
SYMBOL_FFMPEG(retro_get_system_av_info);
2015-09-15 03:22:57 +00:00
SYMBOL_FFMPEG(retro_set_environment);
SYMBOL_FFMPEG(retro_set_video_refresh);
SYMBOL_FFMPEG(retro_set_audio_sample);
SYMBOL_FFMPEG(retro_set_audio_sample_batch);
SYMBOL_FFMPEG(retro_set_input_poll);
SYMBOL_FFMPEG(retro_set_input_state);
2015-09-15 03:22:57 +00:00
SYMBOL_FFMPEG(retro_set_controller_port_device);
2015-09-15 03:22:57 +00:00
SYMBOL_FFMPEG(retro_reset);
SYMBOL_FFMPEG(retro_run);
2015-09-15 03:22:57 +00:00
SYMBOL_FFMPEG(retro_serialize_size);
SYMBOL_FFMPEG(retro_serialize);
SYMBOL_FFMPEG(retro_unserialize);
2015-09-15 03:22:57 +00:00
SYMBOL_FFMPEG(retro_cheat_reset);
SYMBOL_FFMPEG(retro_cheat_set);
2015-09-15 03:22:57 +00:00
SYMBOL_FFMPEG(retro_load_game);
SYMBOL_FFMPEG(retro_load_game_special);
2015-09-15 03:22:57 +00:00
SYMBOL_FFMPEG(retro_unload_game);
SYMBOL_FFMPEG(retro_get_region);
SYMBOL_FFMPEG(retro_get_memory_data);
SYMBOL_FFMPEG(retro_get_memory_size);
#endif
2016-01-25 06:04:54 +00:00
break;
2015-06-28 15:02:01 +00:00
case CORE_TYPE_IMAGEVIEWER:
2015-06-28 16:55:00 +00:00
#ifdef HAVE_IMAGEVIEWER
2015-09-15 03:22:57 +00:00
SYMBOL_IMAGEVIEWER(retro_init);
SYMBOL_IMAGEVIEWER(retro_deinit);
2015-06-28 15:02:01 +00:00
2015-09-15 03:22:57 +00:00
SYMBOL_IMAGEVIEWER(retro_api_version);
SYMBOL_IMAGEVIEWER(retro_get_system_info);
SYMBOL_IMAGEVIEWER(retro_get_system_av_info);
2015-06-28 15:02:01 +00:00
2015-09-15 03:22:57 +00:00
SYMBOL_IMAGEVIEWER(retro_set_environment);
SYMBOL_IMAGEVIEWER(retro_set_video_refresh);
SYMBOL_IMAGEVIEWER(retro_set_audio_sample);
SYMBOL_IMAGEVIEWER(retro_set_audio_sample_batch);
SYMBOL_IMAGEVIEWER(retro_set_input_poll);
SYMBOL_IMAGEVIEWER(retro_set_input_state);
2015-06-28 15:02:01 +00:00
2015-09-15 03:22:57 +00:00
SYMBOL_IMAGEVIEWER(retro_set_controller_port_device);
2015-06-28 15:02:01 +00:00
2015-09-15 03:22:57 +00:00
SYMBOL_IMAGEVIEWER(retro_reset);
SYMBOL_IMAGEVIEWER(retro_run);
2015-06-28 15:02:01 +00:00
2015-09-15 03:22:57 +00:00
SYMBOL_IMAGEVIEWER(retro_serialize_size);
SYMBOL_IMAGEVIEWER(retro_serialize);
SYMBOL_IMAGEVIEWER(retro_unserialize);
2015-06-28 15:02:01 +00:00
2015-09-15 03:22:57 +00:00
SYMBOL_IMAGEVIEWER(retro_cheat_reset);
SYMBOL_IMAGEVIEWER(retro_cheat_set);
2015-06-28 15:02:01 +00:00
2015-09-15 03:22:57 +00:00
SYMBOL_IMAGEVIEWER(retro_load_game);
SYMBOL_IMAGEVIEWER(retro_load_game_special);
2015-06-28 15:02:01 +00:00
2015-09-15 03:22:57 +00:00
SYMBOL_IMAGEVIEWER(retro_unload_game);
SYMBOL_IMAGEVIEWER(retro_get_region);
SYMBOL_IMAGEVIEWER(retro_get_memory_data);
SYMBOL_IMAGEVIEWER(retro_get_memory_size);
2015-06-28 16:55:00 +00:00
#endif
2015-06-28 15:02:01 +00:00
break;
2013-04-28 02:21:52 +00:00
}
2013-04-14 14:24:19 +00:00
}
2015-01-09 20:30:07 +00:00
/**
* libretro_get_current_core_pathname:
* @name : Sanitized name of libretro core.
* @size : Size of @name
*
* Transforms a library id to a name suitable as a pathname.
**/
void libretro_get_current_core_pathname(char *name, size_t size)
{
size_t i;
2015-06-12 15:00:37 +00:00
const char *id = NULL;
2015-01-09 20:30:07 +00:00
struct retro_system_info info = {0};
if (size == 0)
return;
core.retro_get_system_info(&info);
id = info.library_name ? info.library_name :
2015-07-02 12:04:15 +00:00
msg_hash_to_str(MSG_UNKNOWN);
if (!id || strlen(id) >= size)
{
name[0] = '\0';
return;
}
name[strlen(id)] = '\0';
for (i = 0; id[i] != '\0'; i++)
{
char c = id[i];
2015-01-15 01:48:40 +00:00
2015-09-05 12:34:33 +00:00
if (isspace((int)c) || isblank((int)c))
name[i] = '_';
else
2015-09-05 12:34:33 +00:00
name[i] = tolower((int)c);
}
}
2015-01-15 01:48:40 +00:00
/**
* init_libretro_sym:
2015-06-20 21:42:30 +00:00
* @type : Type of core to be loaded.
* If CORE_TYPE_DUMMY, will
2015-06-20 21:42:30 +00:00
* load dummy symbols.
2015-01-15 01:48:40 +00:00
*
* Initializes libretro symbols and
* setups environment callback functions.
**/
2015-06-20 21:42:30 +00:00
void init_libretro_sym(enum rarch_core_type type)
2010-12-30 12:54:49 +00:00
{
2014-09-02 03:57:53 +00:00
/* Guarantee that we can do "dirty" casting.
* Every OS that this program supports should pass this. */
retro_assert(sizeof(void*) == sizeof(void (*)(void)));
2015-06-20 21:42:30 +00:00
load_symbols(type);
2010-12-30 12:54:49 +00:00
}
2015-01-15 01:48:40 +00:00
/**
* uninit_libretro_sym:
*
* Frees libretro core.
*
* Frees all core options,
* associated state, and
* unbind all libretro callback symbols.
**/
2012-04-07 10:17:40 +00:00
void uninit_libretro_sym(void)
2010-12-30 12:54:49 +00:00
{
2011-12-04 16:54:00 +00:00
#ifdef HAVE_DYNAMIC
2010-12-30 12:54:49 +00:00
if (lib_handle)
2011-03-07 18:12:14 +00:00
dylib_close(lib_handle);
2013-04-14 14:24:19 +00:00
lib_handle = NULL;
2011-03-07 18:12:14 +00:00
#endif
2015-12-07 12:56:33 +00:00
core.retro_init = NULL;
core.retro_deinit = NULL;
core.retro_api_version = NULL;
core.retro_get_system_info = NULL;
core.retro_get_system_av_info = NULL;
core.retro_set_environment = NULL;
core.retro_set_video_refresh = NULL;
core.retro_set_audio_sample = NULL;
core.retro_set_audio_sample_batch = NULL;
core.retro_set_input_poll = NULL;
core.retro_set_input_state = NULL;
core.retro_set_controller_port_device = NULL;
2015-12-07 12:56:33 +00:00
core.retro_reset = NULL;
core.retro_run = NULL;
core.retro_serialize_size = NULL;
core.retro_serialize = NULL;
core.retro_unserialize = NULL;
core.retro_cheat_reset = NULL;
core.retro_cheat_set = NULL;
core.retro_load_game = NULL;
core.retro_load_game_special = NULL;
core.retro_unload_game = NULL;
core.retro_get_region = NULL;
core.retro_get_memory_data = NULL;
core.retro_get_memory_size = NULL;
runloop_ctl(RUNLOOP_CTL_SYSTEM_INFO_FREE, NULL);
2015-12-05 13:39:52 +00:00
camera_driver_ctl(RARCH_CAMERA_CTL_UNSET_ACTIVE, NULL);
2015-12-05 14:40:29 +00:00
location_driver_ctl(RARCH_LOCATION_CTL_UNSET_ACTIVE, NULL);
2014-09-02 03:57:53 +00:00
/* Performance counters no longer valid. */
retro_perf_clear();
2011-03-07 18:12:14 +00:00
}
2014-09-02 03:57:53 +00:00
static void rarch_log_libretro(enum retro_log_level level,
const char *fmt, ...)
{
2015-01-09 20:30:07 +00:00
va_list vp;
2015-03-20 18:48:23 +00:00
settings_t *settings = config_get_ptr();
if ((unsigned)level < settings->libretro_log_level)
return;
va_start(vp, fmt);
switch (level)
{
case RETRO_LOG_DEBUG:
RARCH_LOG_V("[libretro DEBUG]", fmt, vp);
break;
case RETRO_LOG_INFO:
RARCH_LOG_OUTPUT_V("[libretro INFO]", fmt, vp);
break;
case RETRO_LOG_WARN:
RARCH_WARN_V("[libretro WARN]", fmt, vp);
break;
case RETRO_LOG_ERROR:
RARCH_ERR_V("[libretro ERROR]", fmt, vp);
break;
default:
break;
}
va_end(vp);
}
2015-01-15 01:48:40 +00:00
/**
* rarch_environment_cb:
* @cmd : Identifier of command.
* @data : Pointer to data.
*
* Environment callback function implementation.
*
* Returns: true (1) if environment callback command could
* be performed, otherwise false (0).
**/
bool rarch_environment_cb(unsigned cmd, void *data)
{
2014-12-05 13:29:40 +00:00
unsigned p;
settings_t *settings = config_get_ptr();
global_t *global = global_get_ptr();
2015-12-10 21:30:25 +00:00
rarch_system_info_t *system = NULL;
runloop_ctl(RUNLOOP_CTL_SYSTEM_INFO_GET, &system);
2014-09-02 03:57:53 +00:00
2014-09-21 12:33:51 +00:00
if (ignore_environment_cb)
return false;
switch (cmd)
{
2012-04-07 09:55:37 +00:00
case RETRO_ENVIRONMENT_GET_OVERSCAN:
2015-03-20 18:48:23 +00:00
*(bool*)data = !settings->video.crop_overscan;
2014-09-02 03:57:53 +00:00
RARCH_LOG("Environ GET_OVERSCAN: %u\n",
2015-03-20 18:48:23 +00:00
(unsigned)!settings->video.crop_overscan);
2011-11-09 21:18:48 +00:00
break;
2012-04-07 09:55:37 +00:00
case RETRO_ENVIRONMENT_GET_CAN_DUPE:
2011-11-22 16:27:02 +00:00
*(bool*)data = true;
2012-04-21 21:25:32 +00:00
RARCH_LOG("Environ GET_CAN_DUPE: true\n");
2011-12-02 18:12:47 +00:00
break;
2012-04-07 09:55:37 +00:00
case RETRO_ENVIRONMENT_GET_VARIABLE:
if (!runloop_ctl(RUNLOOP_CTL_CORE_OPTIONS_GET, data))
{
struct retro_variable *var = (struct retro_variable*)data;
if (var) {
2015-12-26 00:16:33 +00:00
RARCH_LOG("Environ GET_VARIABLE %s: not implemented.\n", var->key);
var->value = NULL;
}
}
break;
2013-04-04 11:58:30 +00:00
case RETRO_ENVIRONMENT_GET_VARIABLE_UPDATE:
*(bool*)data = runloop_ctl(RUNLOOP_CTL_IS_CORE_OPTION_UPDATED, NULL);
2013-04-04 11:58:30 +00:00
break;
2012-04-07 09:55:37 +00:00
case RETRO_ENVIRONMENT_SET_VARIABLES:
2013-04-04 11:58:30 +00:00
RARCH_LOG("Environ SET_VARIABLES.\n");
2012-02-06 15:13:29 +00:00
runloop_ctl(RUNLOOP_CTL_CORE_OPTIONS_DEINIT, NULL);
2015-12-07 13:16:34 +00:00
runloop_ctl(RUNLOOP_CTL_CORE_OPTIONS_INIT, data);
2013-04-04 11:58:30 +00:00
2012-02-06 15:13:29 +00:00
break;
2012-04-07 09:55:37 +00:00
case RETRO_ENVIRONMENT_SET_MESSAGE:
2012-03-11 23:37:44 +00:00
{
2012-04-07 09:55:37 +00:00
const struct retro_message *msg = (const struct retro_message*)data;
2012-04-21 21:25:32 +00:00
RARCH_LOG("Environ SET_MESSAGE: %s\n", msg->msg);
2015-12-07 14:32:14 +00:00
runloop_msg_queue_push(msg->msg, 1, msg->frames, true);
2012-03-11 23:37:44 +00:00
break;
}
2012-04-07 09:55:37 +00:00
case RETRO_ENVIRONMENT_SET_ROTATION:
2012-03-30 17:09:34 +00:00
{
unsigned rotation = *(const unsigned*)data;
2012-04-21 21:25:32 +00:00
RARCH_LOG("Environ SET_ROTATION: %u\n", rotation);
2015-03-20 18:48:23 +00:00
if (!settings->video.allow_rotate)
2012-04-01 14:12:04 +00:00
break;
system->rotation = rotation;
2012-04-01 17:20:37 +00:00
2015-03-22 09:38:26 +00:00
if (!video_driver_set_rotation(rotation))
2012-03-30 17:09:34 +00:00
return false;
break;
}
2012-05-22 18:14:07 +00:00
case RETRO_ENVIRONMENT_SHUTDOWN:
RARCH_LOG("Environ SHUTDOWN.\n");
runloop_ctl(RUNLOOP_CTL_SET_SHUTDOWN, NULL);
2015-11-30 20:42:59 +00:00
runloop_ctl(RUNLOOP_CTL_SET_CORE_SHUTDOWN, NULL);
2012-05-22 18:14:07 +00:00
break;
case RETRO_ENVIRONMENT_SET_PERFORMANCE_LEVEL:
system->performance_level = *(const unsigned*)data;
2014-09-02 03:57:53 +00:00
RARCH_LOG("Environ PERFORMANCE_LEVEL: %u.\n",
system->performance_level);
break;
case RETRO_ENVIRONMENT_GET_SYSTEM_DIRECTORY:
if (string_is_empty(settings->system_directory))
{
char *fullpath = NULL;
2015-11-30 20:42:59 +00:00
runloop_ctl(RUNLOOP_CTL_GET_CONTENT_PATH, &fullpath);
2016-01-26 04:03:41 +00:00
RARCH_WARN("SYSTEM DIR is empty, assume CONTENT DIR %s\n",
fullpath);
fill_pathname_basedir(global->dir.systemdir, fullpath,
sizeof(global->dir.systemdir));
*(const char**)data = global->dir.systemdir;
RARCH_LOG("Environ SYSTEM_DIRECTORY: \"%s\".\n",
global->dir.systemdir);
}
else
{
*(const char**)data = settings->system_directory;
RARCH_LOG("Environ SYSTEM_DIRECTORY: \"%s\".\n",
2015-03-20 18:48:23 +00:00
settings->system_directory);
}
break;
2014-02-07 07:14:02 +00:00
case RETRO_ENVIRONMENT_GET_SAVE_DIRECTORY:
*(const char**)data = rarch_get_current_savefile_dir();
break;
2014-08-01 02:37:00 +00:00
case RETRO_ENVIRONMENT_GET_USERNAME:
2015-03-20 18:48:23 +00:00
*(const char**)data = *settings->username ?
settings->username : NULL;
2014-09-02 03:57:53 +00:00
RARCH_LOG("Environ GET_USERNAME: \"%s\".\n",
2015-03-20 18:48:23 +00:00
settings->username);
2014-08-01 02:37:00 +00:00
break;
case RETRO_ENVIRONMENT_GET_LANGUAGE:
2015-03-20 18:48:23 +00:00
*(unsigned *)data = settings->user_language;
RARCH_LOG("Environ GET_LANGUAGE: \"%u\".\n",
2015-03-20 18:48:23 +00:00
settings->user_language);
break;
2012-06-16 13:07:31 +00:00
case RETRO_ENVIRONMENT_SET_PIXEL_FORMAT:
{
enum retro_pixel_format pix_fmt =
2014-09-02 03:57:53 +00:00
*(const enum retro_pixel_format*)data;
2012-06-16 13:07:31 +00:00
switch (pix_fmt)
{
case RETRO_PIXEL_FORMAT_0RGB1555:
RARCH_LOG("Environ SET_PIXEL_FORMAT: 0RGB1555.\n");
break;
case RETRO_PIXEL_FORMAT_RGB565:
RARCH_LOG("Environ SET_PIXEL_FORMAT: RGB565.\n");
break;
2012-06-16 13:07:31 +00:00
case RETRO_PIXEL_FORMAT_XRGB8888:
RARCH_LOG("Environ SET_PIXEL_FORMAT: XRGB8888.\n");
break;
default:
return false;
}
2015-05-20 18:59:12 +00:00
video_driver_set_pixel_format(pix_fmt);
2012-06-16 13:07:31 +00:00
break;
}
case RETRO_ENVIRONMENT_SET_INPUT_DESCRIPTORS:
{
2016-01-26 04:06:11 +00:00
unsigned retro_id;
2015-01-09 20:30:07 +00:00
const struct retro_input_descriptor *desc = NULL;
2015-06-26 15:46:13 +00:00
static const char *libretro_btn_desc[] = {
"B (bottom)", "Y (left)", "Select", "Start",
"D-Pad Up", "D-Pad Down", "D-Pad Left", "D-Pad Right",
"A (right)", "X (up)",
"L", "R", "L2", "R2", "L3", "R3",
};
memset(system->input_desc_btn, 0,
sizeof(system->input_desc_btn));
2014-09-02 03:57:53 +00:00
2015-01-09 20:30:07 +00:00
desc = (const struct retro_input_descriptor*)data;
for (; desc->description; desc++)
{
2016-01-26 04:06:11 +00:00
unsigned retro_port = desc->port;
retro_id = desc->id;
2015-01-15 01:48:40 +00:00
2015-01-05 00:58:00 +00:00
if (desc->port >= MAX_USERS)
continue;
2014-09-02 03:57:53 +00:00
/* Ignore all others for now. */
if (desc->device != RETRO_DEVICE_JOYPAD &&
desc->device != RETRO_DEVICE_ANALOG)
continue;
2012-10-01 20:15:48 +00:00
if (desc->id >= RARCH_FIRST_CUSTOM_BIND)
continue;
if (desc->device == RETRO_DEVICE_ANALOG)
{
2014-12-05 13:29:40 +00:00
switch (retro_id)
{
case RETRO_DEVICE_ID_ANALOG_X:
switch (desc->index)
{
case RETRO_DEVICE_INDEX_ANALOG_LEFT:
2016-01-26 04:03:41 +00:00
system->input_desc_btn[retro_port]
[RARCH_ANALOG_LEFT_X_PLUS] = desc->description;
system->input_desc_btn[retro_port]
[RARCH_ANALOG_LEFT_X_MINUS] = desc->description;
break;
case RETRO_DEVICE_INDEX_ANALOG_RIGHT:
2016-01-26 04:03:41 +00:00
system->input_desc_btn[retro_port]
[RARCH_ANALOG_RIGHT_X_PLUS] = desc->description;
system->input_desc_btn[retro_port]
[RARCH_ANALOG_RIGHT_X_MINUS] = desc->description;
break;
}
break;
case RETRO_DEVICE_ID_ANALOG_Y:
switch (desc->index)
{
case RETRO_DEVICE_INDEX_ANALOG_LEFT:
2016-01-26 04:03:41 +00:00
system->input_desc_btn[retro_port]
[RARCH_ANALOG_LEFT_Y_PLUS] = desc->description;
system->input_desc_btn[retro_port]
[RARCH_ANALOG_LEFT_Y_MINUS] = desc->description;
break;
case RETRO_DEVICE_INDEX_ANALOG_RIGHT:
2016-01-26 04:03:41 +00:00
system->input_desc_btn[retro_port]
[RARCH_ANALOG_RIGHT_Y_PLUS] = desc->description;
system->input_desc_btn[retro_port]
[RARCH_ANALOG_RIGHT_Y_MINUS] = desc->description;
break;
}
break;
}
}
else
2016-01-26 04:03:41 +00:00
system->input_desc_btn[retro_port]
[retro_id] = desc->description;
}
RARCH_LOG("Environ SET_INPUT_DESCRIPTORS:\n");
2015-03-20 18:48:23 +00:00
for (p = 0; p < settings->input.max_users; p++)
{
2014-12-05 13:29:40 +00:00
for (retro_id = 0; retro_id < RARCH_FIRST_CUSTOM_BIND; retro_id++)
{
const char *description = system->input_desc_btn[p][retro_id];
2015-01-09 20:30:07 +00:00
2015-01-15 01:48:40 +00:00
if (!description)
continue;
RARCH_LOG("\tRetroPad, User %u, Button \"%s\" => \"%s\"\n",
p + 1, libretro_btn_desc[retro_id], description);
}
}
2015-07-27 15:18:10 +00:00
global->has_set.input_descriptors = true;
break;
}
case RETRO_ENVIRONMENT_SET_KEYBOARD_CALLBACK:
{
retro_keyboard_event_t *frontend_key_event = NULL;
retro_keyboard_event_t *key_event = NULL;
const struct retro_keyboard_callback *info =
2014-09-02 03:57:53 +00:00
(const struct retro_keyboard_callback*)data;
2015-01-09 20:30:07 +00:00
runloop_ctl(RUNLOOP_CTL_FRONTEND_KEY_EVENT_GET, &frontend_key_event);
2015-12-13 13:31:06 +00:00
runloop_ctl(RUNLOOP_CTL_KEY_EVENT_GET, &key_event);
2015-01-09 20:30:07 +00:00
RARCH_LOG("Environ SET_KEYBOARD_CALLBACK.\n");
if (key_event)
*key_event = info->callback;
if (frontend_key_event)
*frontend_key_event = *key_event;
break;
}
case RETRO_ENVIRONMENT_SET_DISK_CONTROL_INTERFACE:
RARCH_LOG("Environ SET_DISK_CONTROL_INTERFACE.\n");
system->disk_control =
2014-09-02 03:57:53 +00:00
*(const struct retro_disk_control_callback*)data;
break;
2013-03-27 15:15:15 +00:00
case RETRO_ENVIRONMENT_SET_HW_RENDER:
2014-09-02 03:57:53 +00:00
case RETRO_ENVIRONMENT_SET_HW_RENDER | RETRO_ENVIRONMENT_EXPERIMENTAL:
2013-03-27 15:15:15 +00:00
{
struct retro_hw_render_callback *hw_render = video_driver_callback();
struct retro_hw_render_callback *cb =
2014-09-02 03:57:53 +00:00
(struct retro_hw_render_callback*)data;
2015-01-09 20:30:07 +00:00
RARCH_LOG("Environ SET_HW_RENDER.\n");
switch (cb->context_type)
{
case RETRO_HW_CONTEXT_NONE:
RARCH_LOG("Requesting no HW context.\n");
break;
#if defined(HAVE_OPENGLES2)
case RETRO_HW_CONTEXT_OPENGLES2:
#if defined(HAVE_OPENGLES3)
case RETRO_HW_CONTEXT_OPENGLES3:
#endif
RARCH_LOG("Requesting OpenGLES%u context.\n",
cb->context_type == RETRO_HW_CONTEXT_OPENGLES2 ? 2 : 3);
break;
2014-05-03 13:21:14 +00:00
#if defined(HAVE_OPENGLES3)
case RETRO_HW_CONTEXT_OPENGLES_VERSION:
RARCH_LOG("Requesting OpenGLES%u.%u context.\n",
cb->version_major, cb->version_minor);
break;
2014-06-16 19:02:42 +00:00
#endif
2014-05-03 13:21:14 +00:00
case RETRO_HW_CONTEXT_OPENGL:
case RETRO_HW_CONTEXT_OPENGL_CORE:
2016-01-26 04:03:41 +00:00
RARCH_ERR("Requesting OpenGL context, but RetroArch "
"is compiled against OpenGLES2. Cannot use HW context.\n");
return false;
#elif defined(HAVE_OPENGL)
case RETRO_HW_CONTEXT_OPENGLES2:
case RETRO_HW_CONTEXT_OPENGLES3:
2016-01-26 04:03:41 +00:00
RARCH_ERR("Requesting OpenGLES%u context, but RetroArch "
"is compiled against OpenGL. Cannot use HW context.\n",
cb->context_type == RETRO_HW_CONTEXT_OPENGLES2 ? 2 : 3);
return false;
2014-05-03 13:21:14 +00:00
case RETRO_HW_CONTEXT_OPENGLES_VERSION:
2016-01-26 04:03:41 +00:00
RARCH_ERR("Requesting OpenGLES%u.%u context, but RetroArch "
"is compiled against OpenGL. Cannot use HW context.\n",
2014-05-03 13:21:14 +00:00
cb->version_major, cb->version_minor);
return false;
case RETRO_HW_CONTEXT_OPENGL:
RARCH_LOG("Requesting OpenGL context.\n");
break;
case RETRO_HW_CONTEXT_OPENGL_CORE:
2014-09-02 03:57:53 +00:00
RARCH_LOG("Requesting core OpenGL context (%u.%u).\n",
cb->version_major, cb->version_minor);
break;
#endif
default:
RARCH_LOG("Requesting unknown context.\n");
return false;
}
cb->get_current_framebuffer = video_driver_get_current_framebuffer;
cb->get_proc_address = video_driver_get_proc_address;
2016-01-26 04:03:41 +00:00
/* Old ABI. Don't copy garbage. */
if (cmd & RETRO_ENVIRONMENT_EXPERIMENTAL)
memcpy(hw_render,
2014-09-02 03:57:53 +00:00
cb, offsetof(struct retro_hw_render_callback, stencil));
else
memcpy(hw_render, cb, sizeof(*cb));
2013-03-27 15:15:15 +00:00
break;
}
case RETRO_ENVIRONMENT_SET_SUPPORT_NO_GAME:
{
bool state = *(const bool*)data;
RARCH_LOG("Environ SET_SUPPORT_NO_GAME: %s.\n", state ? "yes" : "no");
system->no_content = state;
break;
}
case RETRO_ENVIRONMENT_GET_LIBRETRO_PATH:
{
const char **path = (const char**)data;
2016-01-25 06:02:26 +00:00
*path = NULL;
#ifdef HAVE_DYNAMIC
2015-03-20 18:48:23 +00:00
*path = settings->libretro;
#endif
break;
}
/* FIXME - PS3 audio driver needs to be fixed so that threaded
* audio works correctly (audio is already on a thread for PS3
2014-09-02 03:57:53 +00:00
* audio driver so that's probably the problem) */
#if defined(HAVE_THREADS) && !defined(__CELLOS_LV2__)
case RETRO_ENVIRONMENT_SET_AUDIO_CALLBACK:
{
const struct retro_audio_callback *info =
2014-09-02 03:57:53 +00:00
(const struct retro_audio_callback*)data;
2015-06-26 15:46:13 +00:00
RARCH_LOG("Environ SET_AUDIO_CALLBACK.\n");
if (recording_driver_get_data_ptr()) /* A/V sync is a must. */
return false;
2013-07-15 18:36:18 +00:00
#ifdef HAVE_NETPLAY
2015-07-27 15:18:10 +00:00
if (global->netplay.enable)
2013-07-15 18:36:18 +00:00
return false;
#endif
2015-05-19 19:35:40 +00:00
audio_driver_set_callback(info);
break;
}
#endif
case RETRO_ENVIRONMENT_SET_FRAME_TIME_CALLBACK:
2013-07-14 11:43:01 +00:00
{
const struct retro_frame_time_callback *info =
2015-01-09 20:30:07 +00:00
(const struct retro_frame_time_callback*)data;
RARCH_LOG("Environ SET_FRAME_TIME_CALLBACK.\n");
2013-07-15 18:36:18 +00:00
#ifdef HAVE_NETPLAY
/* retro_run() will be called in very strange and
2014-09-02 03:57:53 +00:00
* mysterious ways, have to disable it. */
2015-07-27 15:18:10 +00:00
if (global->netplay.enable)
return false;
2013-07-15 18:36:18 +00:00
#endif
system->frame_time = *info;
break;
2013-07-14 11:43:01 +00:00
}
case RETRO_ENVIRONMENT_GET_RUMBLE_INTERFACE:
{
struct retro_rumble_interface *iface =
2014-09-02 03:57:53 +00:00
(struct retro_rumble_interface*)data;
2015-01-09 20:30:07 +00:00
RARCH_LOG("Environ GET_RUMBLE_INTERFACE.\n");
iface->set_rumble_state = input_driver_set_rumble_state;
break;
}
case RETRO_ENVIRONMENT_GET_INPUT_DEVICE_CAPABILITIES:
{
uint64_t *mask = (uint64_t*)data;
2015-01-09 20:30:07 +00:00
RARCH_LOG("Environ GET_INPUT_DEVICE_CAPABILITIES.\n");
if (input_driver_ctl(RARCH_INPUT_CTL_HAS_CAPABILITIES, NULL))
2015-03-23 03:18:27 +00:00
*mask = input_driver_get_capabilities();
else
return false;
break;
}
case RETRO_ENVIRONMENT_GET_SENSOR_INTERFACE:
{
struct retro_sensor_interface *iface =
2014-09-02 03:57:53 +00:00
(struct retro_sensor_interface*)data;
2015-01-09 20:30:07 +00:00
RARCH_LOG("Environ GET_SENSOR_INTERFACE.\n");
2015-01-18 18:28:51 +00:00
iface->set_sensor_state = input_sensor_set_state;
iface->get_sensor_input = input_sensor_get_input;
break;
}
case RETRO_ENVIRONMENT_GET_CAMERA_INTERFACE:
{
2014-09-02 03:57:53 +00:00
struct retro_camera_callback *cb =
(struct retro_camera_callback*)data;
2015-01-09 20:30:07 +00:00
RARCH_LOG("Environ GET_CAMERA_INTERFACE.\n");
cb->start = driver_camera_start;
cb->stop = driver_camera_stop;
system->camera_callback = *cb;
2015-12-05 13:39:52 +00:00
if (cb->caps != 0)
camera_driver_ctl(RARCH_CAMERA_CTL_SET_ACTIVE, NULL);
else
camera_driver_ctl(RARCH_CAMERA_CTL_UNSET_ACTIVE, NULL);
break;
}
case RETRO_ENVIRONMENT_GET_LOCATION_INTERFACE:
{
2014-09-02 03:57:53 +00:00
struct retro_location_callback *cb =
(struct retro_location_callback*)data;
2015-01-09 20:30:07 +00:00
RARCH_LOG("Environ GET_LOCATION_INTERFACE.\n");
cb->start = driver_location_start;
cb->stop = driver_location_stop;
cb->get_position = driver_location_get_position;
cb->set_interval = driver_location_set_interval;
system->location_callback = *cb;
2015-12-05 14:40:29 +00:00
location_driver_ctl(RARCH_LOCATION_CTL_UNSET_ACTIVE, NULL);
break;
}
case RETRO_ENVIRONMENT_GET_LOG_INTERFACE:
{
struct retro_log_callback *cb = (struct retro_log_callback*)data;
2015-01-09 20:30:07 +00:00
RARCH_LOG("Environ GET_LOG_INTERFACE.\n");
cb->log = rarch_log_libretro;
break;
}
case RETRO_ENVIRONMENT_GET_PERF_INTERFACE:
{
struct retro_perf_callback *cb = (struct retro_perf_callback*)data;
2015-01-09 20:30:07 +00:00
RARCH_LOG("Environ GET_PERF_INTERFACE.\n");
2015-09-20 08:02:47 +00:00
cb->get_time_usec = retro_get_time_usec;
cb->get_cpu_features = retro_get_cpu_features;
cb->get_perf_counter = retro_get_perf_counter;
2016-01-26 04:03:41 +00:00
cb->perf_register = retro_perf_register;
2015-09-20 08:02:47 +00:00
cb->perf_start = retro_perf_start;
cb->perf_stop = retro_perf_stop;
2016-01-26 04:03:41 +00:00
cb->perf_log = retro_perf_log;
break;
}
case RETRO_ENVIRONMENT_GET_CORE_ASSETS_DIRECTORY:
{
const char **dir = (const char**)data;
2015-01-09 20:30:07 +00:00
2015-03-20 18:48:23 +00:00
*dir = *settings->core_assets_directory ?
settings->core_assets_directory : NULL;
RARCH_LOG("Environ CORE_ASSETS_DIRECTORY: \"%s\".\n",
2015-03-20 18:48:23 +00:00
settings->core_assets_directory);
break;
}
case RETRO_ENVIRONMENT_SET_SYSTEM_AV_INFO:
{
RARCH_LOG("Environ SET_SYSTEM_AV_INFO.\n");
return driver_ctl(RARCH_DRIVER_CTL_UPDATE_SYSTEM_AV_INFO,
(void**)&data);
}
case RETRO_ENVIRONMENT_SET_SUBSYSTEM_INFO:
2014-04-04 12:58:42 +00:00
{
2014-04-04 14:02:53 +00:00
unsigned i, j;
const struct retro_subsystem_info *info =
2014-09-02 03:57:53 +00:00
(const struct retro_subsystem_info*)data;
2015-01-09 20:30:07 +00:00
RARCH_LOG("Environ SET_SUBSYSTEM_INFO.\n");
2014-04-04 12:58:42 +00:00
for (i = 0; info[i].ident; i++)
2014-04-04 14:02:53 +00:00
{
RARCH_LOG("Special game type: %s\n", info[i].desc);
RARCH_LOG(" Ident: %s\n", info[i].ident);
RARCH_LOG(" ID: %u\n", info[i].id);
2014-07-28 18:01:27 +00:00
RARCH_LOG(" Content:\n");
2014-04-04 14:02:53 +00:00
for (j = 0; j < info[i].num_roms; j++)
{
RARCH_LOG(" %s (%s)\n",
2014-09-02 03:57:53 +00:00
info[i].roms[j].desc, info[i].roms[j].required ?
"required" : "optional");
2014-04-04 14:02:53 +00:00
}
}
2014-04-04 12:58:42 +00:00
free(system->special);
system->special = (struct retro_subsystem_info*)
calloc(i, sizeof(*system->special));
2014-09-02 03:57:53 +00:00
if (!system->special)
2014-04-04 12:58:42 +00:00
return false;
memcpy(system->special, info,
i * sizeof(*system->special));
system->num_special = i;
2014-04-04 12:58:42 +00:00
break;
}
case RETRO_ENVIRONMENT_SET_CONTROLLER_INFO:
{
unsigned i, j;
const struct retro_controller_info *info =
2014-09-02 03:57:53 +00:00
(const struct retro_controller_info*)data;
2015-01-09 20:30:07 +00:00
RARCH_LOG("Environ SET_CONTROLLER_INFO.\n");
for (i = 0; info[i].types; i++)
{
2014-04-12 17:06:37 +00:00
RARCH_LOG("Controller port: %u\n", i + 1);
for (j = 0; j < info[i].num_types; j++)
2014-09-02 03:57:53 +00:00
RARCH_LOG(" %s (ID: %u)\n", info[i].types[j].desc,
info[i].types[j].id);
}
free(system->ports);
system->ports = (struct retro_controller_info*)
calloc(i, sizeof(*system->ports));
if (!system->ports)
return false;
memcpy(system->ports, info,
i * sizeof(*system->ports));
system->num_ports = i;
break;
}
2014-06-07 11:19:28 +00:00
case RETRO_ENVIRONMENT_SET_GEOMETRY:
{
2016-01-26 04:03:41 +00:00
const struct retro_game_geometry *in_geom = NULL;
struct retro_game_geometry *geom = NULL;
struct retro_system_av_info *av_info =
video_viewport_get_system_av_info();
if (av_info)
geom = (struct retro_game_geometry*)&av_info->geometry;
2014-06-07 11:19:28 +00:00
2015-09-28 14:20:26 +00:00
if (!geom)
return false;
2016-01-26 04:03:41 +00:00
in_geom = (const struct retro_game_geometry*)data;
2015-01-09 20:30:07 +00:00
RARCH_LOG("Environ SET_GEOMETRY.\n");
2014-09-02 03:57:53 +00:00
/* Can potentially be called every frame,
* don't do anything unless required. */
2014-06-07 11:19:28 +00:00
if (geom->base_width != in_geom->base_width ||
geom->base_height != in_geom->base_height ||
geom->aspect_ratio != in_geom->aspect_ratio)
{
geom->base_width = in_geom->base_width;
geom->base_height = in_geom->base_height;
geom->aspect_ratio = in_geom->aspect_ratio;
2016-01-26 04:03:41 +00:00
2014-06-07 11:19:28 +00:00
RARCH_LOG("SET_GEOMETRY: %ux%u, aspect: %.3f.\n",
geom->base_width, geom->base_height, geom->aspect_ratio);
/* Forces recomputation of aspect ratios if
2014-09-02 03:57:53 +00:00
* using core-dependent aspect ratios. */
event_cmd_ctl(EVENT_CMD_VIDEO_SET_ASPECT_RATIO, NULL);
2014-09-02 03:57:53 +00:00
/* TODO: Figure out what to do, if anything, with recording. */
2014-06-07 11:19:28 +00:00
}
break;
}
case RETRO_ENVIRONMENT_GET_CURRENT_SOFTWARE_FRAMEBUFFER:
2016-01-26 04:06:11 +00:00
return video_driver_get_current_software_framebuffer(
(struct retro_framebuffer*)data);
2014-09-02 03:57:53 +00:00
/* Private extensions for internal use, not part of libretro API. */
case RETRO_ENVIRONMENT_EXEC:
{
2016-01-26 05:02:08 +00:00
char *fullpath = NULL;
RARCH_LOG("Environ (Private) EXEC.\n");
2015-11-30 20:42:59 +00:00
runloop_ctl(RUNLOOP_CTL_GET_CONTENT_PATH, &fullpath);
if (fullpath != data)
{
2015-11-30 20:42:59 +00:00
runloop_ctl(RUNLOOP_CTL_CLEAR_CONTENT_PATH, NULL);
if (data)
2015-11-30 20:42:59 +00:00
runloop_ctl(RUNLOOP_CTL_SET_CONTENT_PATH, data);
}
#if defined(RARCH_CONSOLE)
2016-01-13 10:16:03 +00:00
frontend_driver_set_fork(true, true, false);
#elif defined(HAVE_DYNAMIC)
rarch_ctl(RARCH_CTL_LOAD_CONTENT, NULL);
#endif
}
break;
default:
2012-04-21 21:25:32 +00:00
RARCH_LOG("Environ UNSUPPORTED (#%u).\n", cmd);
return false;
}
return true;
}