2014-02-07 02:14:02 -05:00
|
|
|
/* RetroArch - A frontend for libretro.
|
2014-01-01 01:50:59 +01:00
|
|
|
* Copyright (C) 2010-2014 - Hans-Kristian Arntzen
|
2015-01-07 17:46:50 +01:00
|
|
|
* Copyright (C) 2011-2015 - Daniel De Matteis
|
2014-04-12 13:59:19 +02:00
|
|
|
*
|
2012-04-21 23:13:50 +02:00
|
|
|
* RetroArch is free software: you can redistribute it and/or modify it under the terms
|
2010-12-30 13:54:49 +01: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 23:13:50 +02:00
|
|
|
* RetroArch is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY;
|
2010-12-30 13:54:49 +01: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 23:31:57 +02:00
|
|
|
* You should have received a copy of the GNU General Public License along with RetroArch.
|
2010-12-30 13:54:49 +01:00
|
|
|
* If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include "dynamic.h"
|
2014-10-21 07:58:58 +02:00
|
|
|
#include <compat/strl.h>
|
|
|
|
#include <compat/posix_string.h>
|
2013-12-09 10:32:12 +01:00
|
|
|
#include "retroarch_logger.h"
|
2013-12-18 19:10:57 +01:00
|
|
|
#include "performance.h"
|
2014-10-22 00:23:06 +02:00
|
|
|
#include <file/file_path.h>
|
2010-12-30 13:54:49 +01:00
|
|
|
#include <string.h>
|
2013-03-22 20:56:23 +01:00
|
|
|
#include <ctype.h>
|
2011-01-07 17:59:53 +01:00
|
|
|
|
|
|
|
#ifdef HAVE_CONFIG_H
|
2010-12-30 13:54:49 +01:00
|
|
|
#include "config.h"
|
2011-01-07 17:59:53 +01:00
|
|
|
#endif
|
|
|
|
|
2014-10-21 05:05:52 +02:00
|
|
|
#include <boolean.h>
|
2013-08-25 11:07:06 +02:00
|
|
|
#include "libretro_private.h"
|
2013-04-14 16:24:19 +02:00
|
|
|
#include "dynamic_dummy.h"
|
2015-01-09 17:40:47 +01:00
|
|
|
#include "retroarch.h"
|
2010-12-30 13:54:49 +01:00
|
|
|
|
2015-01-18 19:28:51 +01:00
|
|
|
#include "input/input_sensor.h"
|
|
|
|
|
2011-12-01 22:36:26 +01:00
|
|
|
#ifdef NEED_DYNAMIC
|
2011-01-19 13:25:18 +01:00
|
|
|
#ifdef _WIN32
|
|
|
|
#include <windows.h>
|
|
|
|
#else
|
|
|
|
#include <dlfcn.h>
|
2011-03-07 19:12:14 +01:00
|
|
|
#endif
|
2011-12-01 21:54:42 +01:00
|
|
|
#endif
|
2011-03-07 19:12:14 +01:00
|
|
|
|
2012-04-05 11:47:43 +02:00
|
|
|
#ifdef HAVE_DYNAMIC
|
2013-07-05 19:29:48 +02:00
|
|
|
#undef SYM
|
2012-04-05 11:47:43 +02:00
|
|
|
#define SYM(x) do { \
|
2012-04-07 11:55:37 +02:00
|
|
|
function_t func = dylib_proc(lib_handle, #x); \
|
2012-04-05 11:47:43 +02:00
|
|
|
memcpy(&p##x, &func, sizeof(func)); \
|
2012-04-21 23:25:32 +02:00
|
|
|
if (p##x == NULL) { RARCH_ERR("Failed to load symbol: \"%s\"\n", #x); rarch_fail(1, "init_libretro_sym()"); } \
|
2011-10-15 12:56:48 +02:00
|
|
|
} while (0)
|
2010-12-30 13:54:49 +01:00
|
|
|
|
2013-04-14 16:24:19 +02:00
|
|
|
static dylib_t lib_handle;
|
2012-04-05 11:47:43 +02:00
|
|
|
#else
|
|
|
|
#define SYM(x) p##x = x
|
2011-01-19 13:25:18 +01:00
|
|
|
#endif
|
2010-12-30 13:54:49 +01:00
|
|
|
|
2013-04-14 16:24:19 +02:00
|
|
|
#define SYM_DUMMY(x) p##x = libretro_dummy_##x
|
|
|
|
|
2012-04-05 11:47:43 +02:00
|
|
|
void (*pretro_init)(void);
|
|
|
|
void (*pretro_deinit)(void);
|
2010-12-30 13:54:49 +01:00
|
|
|
|
2012-04-05 11:47:43 +02:00
|
|
|
unsigned (*pretro_api_version)(void);
|
2010-12-30 13:54:49 +01:00
|
|
|
|
2012-04-05 11:47:43 +02:00
|
|
|
void (*pretro_get_system_info)(struct retro_system_info*);
|
|
|
|
void (*pretro_get_system_av_info)(struct retro_system_av_info*);
|
2010-12-30 13:54:49 +01:00
|
|
|
|
2012-04-05 11:47:43 +02:00
|
|
|
void (*pretro_set_environment)(retro_environment_t);
|
|
|
|
void (*pretro_set_video_refresh)(retro_video_refresh_t);
|
|
|
|
void (*pretro_set_audio_sample)(retro_audio_sample_t);
|
|
|
|
void (*pretro_set_audio_sample_batch)(retro_audio_sample_batch_t);
|
|
|
|
void (*pretro_set_input_poll)(retro_input_poll_t);
|
|
|
|
void (*pretro_set_input_state)(retro_input_state_t);
|
2011-04-17 13:30:59 +02:00
|
|
|
|
2012-04-10 00:37:09 +02:00
|
|
|
void (*pretro_set_controller_port_device)(unsigned, unsigned);
|
2010-12-30 13:54:49 +01:00
|
|
|
|
2012-04-05 11:47:43 +02:00
|
|
|
void (*pretro_reset)(void);
|
|
|
|
void (*pretro_run)(void);
|
2011-01-12 18:05:57 +01:00
|
|
|
|
2012-04-05 11:47:43 +02:00
|
|
|
size_t (*pretro_serialize_size)(void);
|
|
|
|
bool (*pretro_serialize)(void*, size_t);
|
|
|
|
bool (*pretro_unserialize)(const void*, size_t);
|
2010-12-30 13:54:49 +01:00
|
|
|
|
2012-04-05 11:47:43 +02:00
|
|
|
void (*pretro_cheat_reset)(void);
|
|
|
|
void (*pretro_cheat_set)(unsigned, bool, const char*);
|
2011-01-11 16:53:31 +01:00
|
|
|
|
2012-04-07 11:55:37 +02:00
|
|
|
bool (*pretro_load_game)(const struct retro_game_info*);
|
2014-09-02 05:57:53 +02:00
|
|
|
bool (*pretro_load_game_special)(unsigned,
|
|
|
|
const struct retro_game_info*, size_t);
|
2010-12-30 13:54:49 +01:00
|
|
|
|
2012-04-05 11:47:43 +02:00
|
|
|
void (*pretro_unload_game)(void);
|
2010-12-30 13:54:49 +01:00
|
|
|
|
2012-04-05 11:47:43 +02:00
|
|
|
unsigned (*pretro_get_region)(void);
|
2010-12-30 13:54:49 +01:00
|
|
|
|
2012-04-05 11:47:43 +02:00
|
|
|
void *(*pretro_get_memory_data)(unsigned);
|
|
|
|
size_t (*pretro_get_memory_size)(unsigned);
|
2010-12-30 13:54:49 +01:00
|
|
|
|
2014-09-23 03:07:33 +02:00
|
|
|
static bool ignore_environment_cb;
|
|
|
|
|
2012-10-14 20:09:26 +02:00
|
|
|
#ifdef HAVE_DYNAMIC
|
2014-07-28 19:37:25 +02:00
|
|
|
static bool *load_no_content_hook;
|
2014-09-02 05:57:53 +02:00
|
|
|
|
2013-05-02 14:42:58 +02:00
|
|
|
static bool environ_cb_get_system_info(unsigned cmd, void *data)
|
|
|
|
{
|
|
|
|
switch (cmd)
|
|
|
|
{
|
|
|
|
case RETRO_ENVIRONMENT_SET_SUPPORT_NO_GAME:
|
2014-07-28 19:37:25 +02:00
|
|
|
*load_no_content_hook = *(const bool*)data;
|
2013-05-02 14:42:58 +02:00
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2015-01-09 21:30:07 +01: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
|
|
|
|
* 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
|
|
|
|
* avoid overwriting the "real" environ callback.
|
|
|
|
*
|
|
|
|
* For statically linked cores, pass retro_set_environment as argument.
|
|
|
|
*/
|
2014-09-02 05:57:53 +02:00
|
|
|
void libretro_get_environment_info(void (*func)(retro_environment_t),
|
|
|
|
bool *load_no_content)
|
2013-05-02 15:17:31 +02:00
|
|
|
{
|
2014-07-28 19:37:25 +02:00
|
|
|
load_no_content_hook = load_no_content;
|
2013-05-02 15:17:31 +02:00
|
|
|
|
2014-09-09 05:24:32 +02:00
|
|
|
/* load_no_content gets set in this callback. */
|
2013-05-02 15:17:31 +02:00
|
|
|
func(environ_cb_get_system_info);
|
2014-09-21 09:33:51 -03:00
|
|
|
|
|
|
|
/* It's possible that we just set get_system_info callback to the currently running core.
|
|
|
|
* Make sure we reset it to the actual environment callback.
|
|
|
|
* Ignore any environment callbacks here in case we're running on the non-current core. */
|
|
|
|
ignore_environment_cb = true;
|
|
|
|
func(rarch_environment_cb);
|
|
|
|
ignore_environment_cb = false;
|
2013-05-02 15:17:31 +02:00
|
|
|
}
|
|
|
|
|
2014-09-02 05:57:53 +02:00
|
|
|
static dylib_t libretro_get_system_info_lib(const char *path,
|
|
|
|
struct retro_system_info *info, bool *load_no_content)
|
2013-03-17 20:08:24 +01:00
|
|
|
{
|
|
|
|
dylib_t lib = dylib_load(path);
|
|
|
|
if (!lib)
|
|
|
|
return NULL;
|
|
|
|
|
2014-04-12 13:59:19 +02:00
|
|
|
void (*proc)(struct retro_system_info*) =
|
2014-09-02 05:57:53 +02:00
|
|
|
(void (*)(struct retro_system_info*))dylib_proc(lib,
|
|
|
|
"retro_get_system_info");
|
2013-03-17 20:08:24 +01:00
|
|
|
|
|
|
|
if (!proc)
|
|
|
|
{
|
|
|
|
dylib_close(lib);
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
proc(info);
|
2013-05-02 14:42:58 +02:00
|
|
|
|
2014-07-28 19:37:25 +02:00
|
|
|
if (load_no_content)
|
2013-05-02 14:42:58 +02:00
|
|
|
{
|
2014-07-28 19:37:25 +02:00
|
|
|
*load_no_content = false;
|
2013-05-02 14:42:58 +02:00
|
|
|
void (*set_environ)(retro_environment_t) =
|
2014-09-02 05:57:53 +02:00
|
|
|
(void (*)(retro_environment_t))dylib_proc(lib,
|
|
|
|
"retro_set_environment");
|
2013-05-02 14:42:58 +02:00
|
|
|
|
|
|
|
if (!set_environ)
|
|
|
|
return lib;
|
|
|
|
|
2014-07-28 19:37:25 +02:00
|
|
|
libretro_get_environment_info(set_environ, load_no_content);
|
2013-05-02 14:42:58 +02:00
|
|
|
}
|
|
|
|
|
2013-03-17 20:08:24 +01:00
|
|
|
return lib;
|
|
|
|
}
|
|
|
|
|
2015-01-09 21:30:07 +01: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 05:57:53 +02:00
|
|
|
bool libretro_get_system_info(const char *path,
|
|
|
|
struct retro_system_info *info, bool *load_no_content)
|
2013-03-17 20:08:24 +01:00
|
|
|
{
|
|
|
|
struct retro_system_info dummy_info = {0};
|
2014-09-02 05:57:53 +02:00
|
|
|
dylib_t lib = libretro_get_system_info_lib(path,
|
|
|
|
&dummy_info, load_no_content);
|
2013-03-17 20:08:24 +01:00
|
|
|
if (!lib)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
memcpy(info, &dummy_info, sizeof(*info));
|
2013-05-01 00:56:13 +02:00
|
|
|
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 20:08:24 +01:00
|
|
|
dylib_close(lib);
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2015-01-09 21:30:07 +01:00
|
|
|
/**
|
|
|
|
* libretro_free_system_info:
|
|
|
|
* @info : Pointer to system info information.
|
|
|
|
*
|
|
|
|
* Frees system information.
|
|
|
|
**/
|
2013-03-17 20:08:24 +01:00
|
|
|
void libretro_free_system_info(struct retro_system_info *info)
|
|
|
|
{
|
2014-09-15 08:17:16 -04:00
|
|
|
if (!info)
|
|
|
|
return;
|
|
|
|
|
2013-03-17 20:08:24 +01:00
|
|
|
free((void*)info->library_name);
|
|
|
|
free((void*)info->library_version);
|
|
|
|
free((void*)info->valid_extensions);
|
|
|
|
memset(info, 0, sizeof(*info));
|
|
|
|
}
|
2012-10-14 20:09:26 +02:00
|
|
|
#endif
|
2012-10-11 22:31:47 +02:00
|
|
|
|
2014-09-02 05:57:53 +02:00
|
|
|
const struct retro_subsystem_info *libretro_find_subsystem_info(
|
|
|
|
const struct retro_subsystem_info *info, unsigned num_info,
|
2014-04-07 23:02:12 +02:00
|
|
|
const char *ident)
|
|
|
|
{
|
|
|
|
unsigned i;
|
|
|
|
for (i = 0; i < num_info; i++)
|
|
|
|
{
|
|
|
|
if (!strcmp(info[i].ident, ident))
|
|
|
|
return &info[i];
|
2014-09-02 05:57:53 +02:00
|
|
|
else if (!strcmp(info[i].desc, ident))
|
2014-04-07 23:02:12 +02:00
|
|
|
return &info[i];
|
|
|
|
}
|
|
|
|
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
2015-01-15 02:48:40 +01:00
|
|
|
/**
|
|
|
|
* 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.
|
|
|
|
**/
|
2014-09-02 05:57:53 +02:00
|
|
|
const struct retro_controller_description *
|
|
|
|
libretro_find_controller_description(
|
|
|
|
const struct retro_controller_info *info, unsigned id)
|
2014-04-12 13:22:24 +02:00
|
|
|
{
|
|
|
|
unsigned i;
|
2015-01-09 21:30:07 +01:00
|
|
|
|
2014-04-12 13:22:24 +02:00
|
|
|
for (i = 0; i < info->num_types; i++)
|
|
|
|
{
|
2015-01-15 02:48:40 +01:00
|
|
|
if (info->types[i].id != id)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
return &info->types[i];
|
2014-04-12 13:22:24 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
2015-01-15 02:48:40 +01:00
|
|
|
/**
|
|
|
|
* load_symbols:
|
|
|
|
* @dummy : Load dummy symbols if true
|
|
|
|
*
|
|
|
|
* Setup libretro callback symbols.
|
|
|
|
**/
|
2013-04-28 04:21:52 +02:00
|
|
|
static void load_symbols(bool is_dummy)
|
2010-12-30 13:54:49 +01:00
|
|
|
{
|
2013-04-28 04:21:52 +02:00
|
|
|
if (is_dummy)
|
2012-10-11 22:31:47 +02:00
|
|
|
{
|
2013-04-28 04:21:52 +02:00
|
|
|
SYM_DUMMY(retro_init);
|
|
|
|
SYM_DUMMY(retro_deinit);
|
2012-10-11 22:31:47 +02:00
|
|
|
|
2013-04-28 04:21:52 +02:00
|
|
|
SYM_DUMMY(retro_api_version);
|
|
|
|
SYM_DUMMY(retro_get_system_info);
|
|
|
|
SYM_DUMMY(retro_get_system_av_info);
|
2010-12-30 13:54:49 +01:00
|
|
|
|
2013-04-28 04:21:52 +02:00
|
|
|
SYM_DUMMY(retro_set_environment);
|
|
|
|
SYM_DUMMY(retro_set_video_refresh);
|
|
|
|
SYM_DUMMY(retro_set_audio_sample);
|
|
|
|
SYM_DUMMY(retro_set_audio_sample_batch);
|
|
|
|
SYM_DUMMY(retro_set_input_poll);
|
|
|
|
SYM_DUMMY(retro_set_input_state);
|
2010-12-30 13:54:49 +01:00
|
|
|
|
2013-04-28 04:21:52 +02:00
|
|
|
SYM_DUMMY(retro_set_controller_port_device);
|
2012-04-05 11:47:43 +02:00
|
|
|
|
2013-04-28 04:21:52 +02:00
|
|
|
SYM_DUMMY(retro_reset);
|
|
|
|
SYM_DUMMY(retro_run);
|
2012-04-05 11:47:43 +02:00
|
|
|
|
2013-04-28 04:21:52 +02:00
|
|
|
SYM_DUMMY(retro_serialize_size);
|
|
|
|
SYM_DUMMY(retro_serialize);
|
|
|
|
SYM_DUMMY(retro_unserialize);
|
2012-04-05 11:47:43 +02:00
|
|
|
|
2013-04-28 04:21:52 +02:00
|
|
|
SYM_DUMMY(retro_cheat_reset);
|
|
|
|
SYM_DUMMY(retro_cheat_set);
|
2012-04-05 11:47:43 +02:00
|
|
|
|
2013-04-28 04:21:52 +02:00
|
|
|
SYM_DUMMY(retro_load_game);
|
|
|
|
SYM_DUMMY(retro_load_game_special);
|
2012-04-05 11:47:43 +02:00
|
|
|
|
2013-04-28 04:21:52 +02:00
|
|
|
SYM_DUMMY(retro_unload_game);
|
|
|
|
SYM_DUMMY(retro_get_region);
|
|
|
|
SYM_DUMMY(retro_get_memory_data);
|
|
|
|
SYM_DUMMY(retro_get_memory_size);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
#ifdef HAVE_DYNAMIC
|
2014-09-02 05:57:53 +02:00
|
|
|
/* 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(g_settings.libretro,
|
|
|
|
sizeof(g_settings.libretro));
|
|
|
|
|
|
|
|
RARCH_LOG("Loading dynamic libretro from: \"%s\"\n",
|
|
|
|
g_settings.libretro);
|
2013-05-04 14:32:00 +02:00
|
|
|
lib_handle = dylib_load(g_settings.libretro);
|
2013-04-28 04:21:52 +02:00
|
|
|
if (!lib_handle)
|
|
|
|
{
|
2014-09-02 05:57:53 +02:00
|
|
|
RARCH_ERR("Failed to open dynamic library: \"%s\"\n",
|
|
|
|
g_settings.libretro);
|
2013-04-28 04:21:52 +02:00
|
|
|
rarch_fail(1, "load_dynamic()");
|
|
|
|
}
|
|
|
|
#endif
|
2010-12-30 13:54:49 +01:00
|
|
|
|
2013-04-28 04:21:52 +02:00
|
|
|
SYM(retro_init);
|
|
|
|
SYM(retro_deinit);
|
2013-04-14 16:24:19 +02:00
|
|
|
|
2013-04-28 04:21:52 +02:00
|
|
|
SYM(retro_api_version);
|
|
|
|
SYM(retro_get_system_info);
|
|
|
|
SYM(retro_get_system_av_info);
|
2013-04-14 16:24:19 +02:00
|
|
|
|
2013-04-28 04:21:52 +02:00
|
|
|
SYM(retro_set_environment);
|
|
|
|
SYM(retro_set_video_refresh);
|
|
|
|
SYM(retro_set_audio_sample);
|
|
|
|
SYM(retro_set_audio_sample_batch);
|
|
|
|
SYM(retro_set_input_poll);
|
|
|
|
SYM(retro_set_input_state);
|
2013-04-14 16:24:19 +02:00
|
|
|
|
2013-04-28 04:21:52 +02:00
|
|
|
SYM(retro_set_controller_port_device);
|
2013-04-14 16:24:19 +02:00
|
|
|
|
2013-04-28 04:21:52 +02:00
|
|
|
SYM(retro_reset);
|
|
|
|
SYM(retro_run);
|
2013-04-14 16:24:19 +02:00
|
|
|
|
2013-04-28 04:21:52 +02:00
|
|
|
SYM(retro_serialize_size);
|
|
|
|
SYM(retro_serialize);
|
|
|
|
SYM(retro_unserialize);
|
2013-04-14 16:24:19 +02:00
|
|
|
|
2013-04-28 04:21:52 +02:00
|
|
|
SYM(retro_cheat_reset);
|
|
|
|
SYM(retro_cheat_set);
|
2013-04-14 16:24:19 +02:00
|
|
|
|
2013-04-28 04:21:52 +02:00
|
|
|
SYM(retro_load_game);
|
|
|
|
SYM(retro_load_game_special);
|
2013-04-14 16:24:19 +02:00
|
|
|
|
2013-04-28 04:21:52 +02:00
|
|
|
SYM(retro_unload_game);
|
|
|
|
SYM(retro_get_region);
|
|
|
|
SYM(retro_get_memory_data);
|
|
|
|
SYM(retro_get_memory_size);
|
|
|
|
}
|
2013-04-14 16:24:19 +02:00
|
|
|
}
|
|
|
|
|
2015-01-09 21:30:07 +01: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.
|
|
|
|
**/
|
2013-03-22 23:42:34 +01:00
|
|
|
void libretro_get_current_core_pathname(char *name, size_t size)
|
2013-03-22 20:54:33 +01:00
|
|
|
{
|
2013-10-22 15:08:17 +02:00
|
|
|
size_t i;
|
2015-01-09 21:30:07 +01:00
|
|
|
const char *id = NULL;
|
|
|
|
struct retro_system_info info = {0};
|
|
|
|
|
2013-03-22 20:54:33 +01:00
|
|
|
if (size == 0)
|
|
|
|
return;
|
|
|
|
|
2013-03-22 20:56:23 +01:00
|
|
|
pretro_get_system_info(&info);
|
2015-01-09 21:30:07 +01:00
|
|
|
id = info.library_name ? info.library_name : "Unknown";
|
2013-03-22 20:54:33 +01:00
|
|
|
|
|
|
|
if (!id || strlen(id) >= size)
|
|
|
|
{
|
|
|
|
name[0] = '\0';
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
name[strlen(id)] = '\0';
|
|
|
|
|
2013-10-22 15:08:17 +02:00
|
|
|
for (i = 0; id[i] != '\0'; i++)
|
2013-03-22 20:54:33 +01:00
|
|
|
{
|
|
|
|
char c = id[i];
|
2015-01-15 02:48:40 +01:00
|
|
|
|
2013-03-22 20:54:33 +01:00
|
|
|
if (isspace(c) || isblank(c))
|
|
|
|
name[i] = '_';
|
|
|
|
else
|
|
|
|
name[i] = tolower(c);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-01-15 02:48:40 +01:00
|
|
|
/**
|
|
|
|
* init_libretro_sym:
|
|
|
|
* @dummy : Load dummy symbols if true
|
|
|
|
*
|
|
|
|
* Initializes libretro symbols and
|
|
|
|
* setups environment callback functions.
|
|
|
|
**/
|
2013-04-14 16:24:19 +02:00
|
|
|
void init_libretro_sym(bool dummy)
|
2010-12-30 13:54:49 +01:00
|
|
|
{
|
2014-09-02 05:57:53 +02:00
|
|
|
/* Guarantee that we can do "dirty" casting.
|
|
|
|
* Every OS that this program supports should pass this. */
|
2012-04-21 23:25:32 +02:00
|
|
|
rarch_assert(sizeof(void*) == sizeof(void (*)(void)));
|
2011-05-31 15:03:59 +02:00
|
|
|
|
2013-04-28 04:21:52 +02:00
|
|
|
if (!dummy)
|
2011-08-23 17:10:02 +02:00
|
|
|
{
|
2013-04-14 16:24:19 +02:00
|
|
|
#ifdef HAVE_DYNAMIC
|
2014-09-02 05:57:53 +02:00
|
|
|
/* Try to verify that -lretro was not linked in from other modules
|
|
|
|
* since loading it dynamically and with -l will fail hard. */
|
2013-04-14 16:24:19 +02:00
|
|
|
function_t sym = dylib_proc(NULL, "retro_init");
|
|
|
|
if (sym)
|
|
|
|
{
|
2014-04-12 13:59:19 +02:00
|
|
|
RARCH_ERR("Serious problem. RetroArch wants to load libretro dyamically, but it is already linked.\n");
|
2013-04-14 16:24:19 +02:00
|
|
|
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");
|
|
|
|
rarch_fail(1, "init_libretro_sym()");
|
|
|
|
}
|
2011-08-23 17:10:02 +02:00
|
|
|
|
2013-04-14 16:24:19 +02:00
|
|
|
if (!*g_settings.libretro)
|
|
|
|
{
|
|
|
|
RARCH_ERR("RetroArch is built for dynamic libretro, but libretro_path is not set. Cannot continue.\n");
|
|
|
|
rarch_fail(1, "init_libretro_sym()");
|
|
|
|
}
|
2010-12-30 14:26:12 +01:00
|
|
|
#endif
|
2013-04-14 16:24:19 +02:00
|
|
|
}
|
|
|
|
|
2013-04-28 04:21:52 +02:00
|
|
|
load_symbols(dummy);
|
|
|
|
|
2013-08-24 22:35:54 +02:00
|
|
|
pretro_set_environment(rarch_environment_cb);
|
2010-12-30 13:54:49 +01:00
|
|
|
}
|
|
|
|
|
2015-01-15 02:48:40 +01:00
|
|
|
/**
|
|
|
|
* uninit_libretro_sym:
|
|
|
|
*
|
|
|
|
* Frees libretro core.
|
|
|
|
*
|
|
|
|
* Frees all core options,
|
|
|
|
* associated state, and
|
|
|
|
* unbind all libretro callback symbols.
|
|
|
|
**/
|
2012-04-07 12:17:40 +02:00
|
|
|
void uninit_libretro_sym(void)
|
2010-12-30 13:54:49 +01:00
|
|
|
{
|
2011-12-04 17:54:00 +01:00
|
|
|
#ifdef HAVE_DYNAMIC
|
2010-12-30 13:54:49 +01:00
|
|
|
if (lib_handle)
|
2011-03-07 19:12:14 +01:00
|
|
|
dylib_close(lib_handle);
|
2013-04-14 16:24:19 +02:00
|
|
|
lib_handle = NULL;
|
2011-03-07 19:12:14 +01:00
|
|
|
#endif
|
2013-04-22 21:10:17 +02:00
|
|
|
|
2013-05-07 01:52:53 +02:00
|
|
|
if (g_extern.system.core_options)
|
|
|
|
{
|
|
|
|
core_option_flush(g_extern.system.core_options);
|
|
|
|
core_option_free(g_extern.system.core_options);
|
|
|
|
}
|
|
|
|
|
2014-09-02 05:57:53 +02:00
|
|
|
/* No longer valid. */
|
2014-04-04 14:58:42 +02:00
|
|
|
free(g_extern.system.special);
|
2014-04-12 13:22:24 +02:00
|
|
|
free(g_extern.system.ports);
|
2013-05-07 01:52:53 +02:00
|
|
|
memset(&g_extern.system, 0, sizeof(g_extern.system));
|
2014-10-01 22:34:48 +02:00
|
|
|
driver.camera_active = false;
|
|
|
|
driver.location_active = false;
|
2013-12-18 19:10:57 +01:00
|
|
|
|
2014-09-02 05:57:53 +02:00
|
|
|
/* Performance counters no longer valid. */
|
2013-12-18 19:10:57 +01:00
|
|
|
retro_perf_clear();
|
2011-03-07 19:12:14 +01:00
|
|
|
}
|
|
|
|
|
2011-12-01 22:36:26 +01:00
|
|
|
#ifdef NEED_DYNAMIC
|
2015-01-09 21:30:07 +01:00
|
|
|
/**
|
|
|
|
* dylib_load:
|
|
|
|
* @path : Path to libretro core library.
|
|
|
|
*
|
|
|
|
* Platform independent dylib loading.
|
|
|
|
*
|
|
|
|
* Returns: library handle on success, otherwise NULL.
|
|
|
|
**/
|
2011-03-07 19:12:14 +01:00
|
|
|
dylib_t dylib_load(const char *path)
|
|
|
|
{
|
2011-01-19 13:25:18 +01:00
|
|
|
#ifdef _WIN32
|
2012-03-05 21:39:27 +01:00
|
|
|
dylib_t lib = LoadLibrary(path);
|
|
|
|
if (!lib)
|
2014-09-02 05:57:53 +02:00
|
|
|
RARCH_ERR("Failed to load library, error code: 0x%x\n",
|
|
|
|
(unsigned)GetLastError());
|
2012-03-05 21:39:27 +01:00
|
|
|
return lib;
|
2011-01-19 13:25:18 +01:00
|
|
|
#else
|
2012-10-03 23:33:17 +02:00
|
|
|
dylib_t lib = dlopen(path, RTLD_LAZY);
|
|
|
|
if (!lib)
|
|
|
|
RARCH_ERR("dylib_load() failed: \"%s\".\n", dlerror());
|
|
|
|
return lib;
|
2011-01-19 13:25:18 +01:00
|
|
|
#endif
|
2011-03-07 19:12:14 +01:00
|
|
|
}
|
|
|
|
|
2011-05-31 15:03:59 +02:00
|
|
|
function_t dylib_proc(dylib_t lib, const char *proc)
|
2011-03-07 19:12:14 +01:00
|
|
|
{
|
2015-01-09 21:30:07 +01:00
|
|
|
function_t sym;
|
|
|
|
void *ptr_sym = NULL;
|
|
|
|
|
|
|
|
(void)ptr_sym;
|
|
|
|
|
2011-03-07 19:12:14 +01:00
|
|
|
#ifdef _WIN32
|
2015-01-09 21:30:07 +01:00
|
|
|
sym = (function_t)GetProcAddress(lib ?
|
2014-09-02 05:57:53 +02:00
|
|
|
(HMODULE)lib : GetModuleHandle(NULL), proc);
|
2011-03-07 19:12:14 +01:00
|
|
|
#else
|
2011-10-27 23:40:34 +02:00
|
|
|
if (lib)
|
2014-04-12 13:59:19 +02:00
|
|
|
ptr_sym = dlsym(lib, proc);
|
2011-10-27 23:40:34 +02:00
|
|
|
else
|
|
|
|
{
|
|
|
|
void *handle = dlopen(NULL, RTLD_LAZY);
|
|
|
|
if (handle)
|
|
|
|
{
|
|
|
|
ptr_sym = dlsym(handle, proc);
|
|
|
|
dlclose(handle);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-09-02 05:57:53 +02:00
|
|
|
/* Dirty hack to workaround the non-legality of
|
|
|
|
* (void*) -> fn-pointer casts. */
|
2011-05-31 15:03:59 +02:00
|
|
|
memcpy(&sym, &ptr_sym, sizeof(void*));
|
2011-03-07 19:12:14 +01:00
|
|
|
#endif
|
2011-03-07 19:56:40 +01:00
|
|
|
|
|
|
|
return sym;
|
2011-03-07 19:12:14 +01:00
|
|
|
}
|
|
|
|
|
2015-01-09 21:30:07 +01:00
|
|
|
/**
|
|
|
|
* dylib_close:
|
|
|
|
* @lib : Library handle.
|
|
|
|
*
|
|
|
|
* Frees library handle.
|
|
|
|
**/
|
2011-03-07 19:12:14 +01:00
|
|
|
void dylib_close(dylib_t lib)
|
|
|
|
{
|
|
|
|
#ifdef _WIN32
|
2011-12-24 13:46:12 +01:00
|
|
|
FreeLibrary((HMODULE)lib);
|
2011-03-07 19:12:14 +01:00
|
|
|
#else
|
2014-09-02 21:12:24 +01:00
|
|
|
#ifndef NO_DLCLOSE
|
2011-03-07 19:12:14 +01:00
|
|
|
dlclose(lib);
|
2010-12-30 13:54:49 +01:00
|
|
|
#endif
|
2014-09-02 21:12:24 +01:00
|
|
|
#endif
|
2010-12-30 13:54:49 +01:00
|
|
|
}
|
2012-01-02 15:42:53 +01:00
|
|
|
#endif
|
2011-03-07 19:12:14 +01:00
|
|
|
|
2014-09-02 05:57:53 +02:00
|
|
|
static void rarch_log_libretro(enum retro_log_level level,
|
|
|
|
const char *fmt, ...)
|
2013-12-09 10:32:12 +01:00
|
|
|
{
|
2015-01-09 21:30:07 +01:00
|
|
|
va_list vp;
|
2014-03-01 12:02:48 +01:00
|
|
|
if ((unsigned)level < g_settings.libretro_log_level)
|
|
|
|
return;
|
|
|
|
|
2013-12-09 10:32:12 +01:00
|
|
|
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 02:48:40 +01: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).
|
|
|
|
**/
|
2013-08-24 22:35:54 +02:00
|
|
|
bool rarch_environment_cb(unsigned cmd, void *data)
|
2011-10-27 23:40:34 +02:00
|
|
|
{
|
2014-12-05 14:29:40 +01:00
|
|
|
unsigned p;
|
2014-09-02 05:57:53 +02:00
|
|
|
|
2014-09-21 09:33:51 -03:00
|
|
|
if (ignore_environment_cb)
|
|
|
|
return false;
|
|
|
|
|
2011-10-27 23:40:34 +02:00
|
|
|
switch (cmd)
|
|
|
|
{
|
2012-04-07 11:55:37 +02:00
|
|
|
case RETRO_ENVIRONMENT_GET_OVERSCAN:
|
2011-11-09 22:18:48 +01:00
|
|
|
*(bool*)data = !g_settings.video.crop_overscan;
|
2014-09-02 05:57:53 +02:00
|
|
|
RARCH_LOG("Environ GET_OVERSCAN: %u\n",
|
|
|
|
(unsigned)!g_settings.video.crop_overscan);
|
2011-11-09 22:18:48 +01:00
|
|
|
break;
|
|
|
|
|
2012-04-07 11:55:37 +02:00
|
|
|
case RETRO_ENVIRONMENT_GET_CAN_DUPE:
|
2011-11-22 17:27:02 +01:00
|
|
|
*(bool*)data = true;
|
2012-04-21 23:25:32 +02:00
|
|
|
RARCH_LOG("Environ GET_CAN_DUPE: true\n");
|
2011-12-02 19:12:47 +01:00
|
|
|
break;
|
|
|
|
|
2012-04-07 11:55:37 +02:00
|
|
|
case RETRO_ENVIRONMENT_GET_VARIABLE:
|
2012-02-06 15:51:35 +01:00
|
|
|
{
|
2012-04-07 11:55:37 +02:00
|
|
|
struct retro_variable *var = (struct retro_variable*)data;
|
2013-04-04 13:58:30 +02:00
|
|
|
RARCH_LOG("Environ GET_VARIABLE %s:\n", var->key);
|
2012-02-06 15:51:35 +01:00
|
|
|
|
2013-04-04 13:58:30 +02:00
|
|
|
if (g_extern.system.core_options)
|
|
|
|
core_option_get(g_extern.system.core_options, var);
|
|
|
|
else
|
|
|
|
var->value = NULL;
|
2012-02-06 15:51:35 +01:00
|
|
|
|
2013-04-15 16:03:44 +02:00
|
|
|
RARCH_LOG("\t%s\n", var->value ? var->value : "N/A");
|
2012-02-06 15:51:35 +01:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2013-04-04 13:58:30 +02:00
|
|
|
case RETRO_ENVIRONMENT_GET_VARIABLE_UPDATE:
|
|
|
|
*(bool*)data = g_extern.system.core_options ?
|
|
|
|
core_option_updated(g_extern.system.core_options) : false;
|
|
|
|
break;
|
|
|
|
|
2012-04-07 11:55:37 +02:00
|
|
|
case RETRO_ENVIRONMENT_SET_VARIABLES:
|
2012-02-06 16:13:29 +01:00
|
|
|
{
|
2013-04-04 13:58:30 +02:00
|
|
|
RARCH_LOG("Environ SET_VARIABLES.\n");
|
2012-02-06 16:13:29 +01:00
|
|
|
|
2013-04-04 13:58:30 +02:00
|
|
|
if (g_extern.system.core_options)
|
|
|
|
{
|
|
|
|
core_option_flush(g_extern.system.core_options);
|
|
|
|
core_option_free(g_extern.system.core_options);
|
2012-02-06 16:13:29 +01:00
|
|
|
}
|
2013-04-04 13:58:30 +02:00
|
|
|
|
2014-09-02 05:57:53 +02:00
|
|
|
const struct retro_variable *vars =
|
|
|
|
(const struct retro_variable*)data;
|
2013-05-12 21:21:19 +02:00
|
|
|
|
|
|
|
const char *options_path = g_settings.core_options_path;
|
2015-01-07 20:42:36 +01:00
|
|
|
char buf[PATH_MAX_LENGTH];
|
2013-05-12 21:21:19 +02:00
|
|
|
if (!*options_path && *g_extern.config_path)
|
|
|
|
{
|
2014-09-02 05:57:53 +02:00
|
|
|
fill_pathname_resolve_relative(buf, g_extern.config_path,
|
|
|
|
"retroarch-core-options.cfg", sizeof(buf));
|
2013-05-12 21:21:19 +02:00
|
|
|
options_path = buf;
|
|
|
|
}
|
|
|
|
g_extern.system.core_options = core_option_new(options_path, vars);
|
2013-04-04 13:58:30 +02:00
|
|
|
|
2012-02-06 16:13:29 +01:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2012-04-07 11:55:37 +02:00
|
|
|
case RETRO_ENVIRONMENT_SET_MESSAGE:
|
2012-03-12 00:37:44 +01:00
|
|
|
{
|
2012-04-07 11:55:37 +02:00
|
|
|
const struct retro_message *msg = (const struct retro_message*)data;
|
2012-04-21 23:25:32 +02:00
|
|
|
RARCH_LOG("Environ SET_MESSAGE: %s\n", msg->msg);
|
2012-03-12 00:37:44 +01:00
|
|
|
if (g_extern.msg_queue)
|
2013-06-16 11:04:41 +02:00
|
|
|
{
|
|
|
|
msg_queue_clear(g_extern.msg_queue);
|
2012-03-12 00:37:44 +01:00
|
|
|
msg_queue_push(g_extern.msg_queue, msg->msg, 1, msg->frames);
|
2013-06-16 11:04:41 +02:00
|
|
|
}
|
2012-03-12 00:37:44 +01:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2012-04-07 11:55:37 +02:00
|
|
|
case RETRO_ENVIRONMENT_SET_ROTATION:
|
2012-03-30 19:09:34 +02:00
|
|
|
{
|
|
|
|
unsigned rotation = *(const unsigned*)data;
|
2012-04-21 23:25:32 +02:00
|
|
|
RARCH_LOG("Environ SET_ROTATION: %u\n", rotation);
|
2012-04-01 16:12:04 +02:00
|
|
|
if (!g_settings.video.allow_rotate)
|
|
|
|
break;
|
|
|
|
|
2012-04-01 19:20:37 +02:00
|
|
|
g_extern.system.rotation = rotation;
|
|
|
|
|
2012-04-01 19:38:50 +02:00
|
|
|
if (driver.video && driver.video->set_rotation)
|
|
|
|
{
|
|
|
|
if (driver.video_data)
|
2014-08-15 20:51:59 +02:00
|
|
|
driver.video->set_rotation(driver.video_data, rotation);
|
2012-04-01 19:38:50 +02:00
|
|
|
}
|
2012-03-30 19:09:34 +02:00
|
|
|
else
|
|
|
|
return false;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2012-05-22 20:14:07 +02:00
|
|
|
case RETRO_ENVIRONMENT_SHUTDOWN:
|
2012-06-12 00:28:34 +02:00
|
|
|
RARCH_LOG("Environ SHUTDOWN.\n");
|
2012-05-22 20:14:07 +02:00
|
|
|
g_extern.system.shutdown = true;
|
2014-09-21 09:18:26 +02:00
|
|
|
g_extern.core_shutdown_initiated = true;
|
2012-05-22 20:14:07 +02:00
|
|
|
break;
|
|
|
|
|
2012-06-01 16:50:38 +02:00
|
|
|
case RETRO_ENVIRONMENT_SET_PERFORMANCE_LEVEL:
|
|
|
|
g_extern.system.performance_level = *(const unsigned*)data;
|
2014-09-02 05:57:53 +02:00
|
|
|
RARCH_LOG("Environ PERFORMANCE_LEVEL: %u.\n",
|
|
|
|
g_extern.system.performance_level);
|
2012-06-12 00:28:34 +02:00
|
|
|
break;
|
|
|
|
|
|
|
|
case RETRO_ENVIRONMENT_GET_SYSTEM_DIRECTORY:
|
2014-09-02 05:57:53 +02:00
|
|
|
*(const char**)data = *g_settings.system_directory ?
|
|
|
|
g_settings.system_directory : NULL;
|
|
|
|
RARCH_LOG("Environ SYSTEM_DIRECTORY: \"%s\".\n",
|
|
|
|
g_settings.system_directory);
|
2012-06-01 16:50:38 +02:00
|
|
|
break;
|
|
|
|
|
2014-02-07 02:14:02 -05:00
|
|
|
case RETRO_ENVIRONMENT_GET_SAVE_DIRECTORY:
|
2014-09-02 05:57:53 +02:00
|
|
|
*(const char**)data = *g_extern.savefile_dir ?
|
|
|
|
g_extern.savefile_dir : NULL;
|
|
|
|
RARCH_LOG("Environ SAVE_DIRECTORY: \"%s\".\n",
|
|
|
|
g_extern.savefile_dir);
|
2014-04-12 13:59:19 +02:00
|
|
|
break;
|
2014-02-07 01:56:41 -05:00
|
|
|
|
2014-08-01 04:37:00 +02:00
|
|
|
case RETRO_ENVIRONMENT_GET_USERNAME:
|
2014-09-02 05:57:53 +02:00
|
|
|
*(const char**)data = *g_settings.username ?
|
|
|
|
g_settings.username : NULL;
|
|
|
|
RARCH_LOG("Environ GET_USERNAME: \"%s\".\n",
|
|
|
|
g_settings.username);
|
2014-08-01 04:37:00 +02:00
|
|
|
break;
|
|
|
|
|
2014-08-01 07:45:56 +02:00
|
|
|
case RETRO_ENVIRONMENT_GET_LANGUAGE:
|
|
|
|
*(unsigned *)data = g_settings.user_language;
|
2014-09-05 19:43:28 -04:00
|
|
|
RARCH_LOG("Environ GET_LANGUAGE: \"%u\".\n",
|
2014-09-02 05:57:53 +02:00
|
|
|
g_settings.user_language);
|
2014-08-01 07:45:56 +02:00
|
|
|
break;
|
|
|
|
|
2012-06-16 15:07:31 +02:00
|
|
|
case RETRO_ENVIRONMENT_SET_PIXEL_FORMAT:
|
|
|
|
{
|
2014-09-02 05:57:53 +02:00
|
|
|
enum retro_pixel_format pix_fmt =
|
|
|
|
*(const enum retro_pixel_format*)data;
|
|
|
|
|
2012-06-16 15:07:31 +02:00
|
|
|
switch (pix_fmt)
|
|
|
|
{
|
|
|
|
case RETRO_PIXEL_FORMAT_0RGB1555:
|
|
|
|
RARCH_LOG("Environ SET_PIXEL_FORMAT: 0RGB1555.\n");
|
|
|
|
break;
|
|
|
|
|
2012-10-20 01:12:02 +02:00
|
|
|
case RETRO_PIXEL_FORMAT_RGB565:
|
|
|
|
RARCH_LOG("Environ SET_PIXEL_FORMAT: RGB565.\n");
|
|
|
|
break;
|
2012-06-16 15:07:31 +02:00
|
|
|
case RETRO_PIXEL_FORMAT_XRGB8888:
|
|
|
|
RARCH_LOG("Environ SET_PIXEL_FORMAT: XRGB8888.\n");
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
return false;
|
|
|
|
}
|
2014-04-12 13:59:19 +02:00
|
|
|
|
2012-10-20 01:12:02 +02:00
|
|
|
g_extern.system.pix_fmt = pix_fmt;
|
2012-06-16 15:07:31 +02:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2012-09-09 23:35:23 +02:00
|
|
|
case RETRO_ENVIRONMENT_SET_INPUT_DESCRIPTORS:
|
|
|
|
{
|
2014-12-05 14:29:40 +01:00
|
|
|
unsigned retro_id, retro_port;
|
2015-01-09 21:30:07 +01:00
|
|
|
const struct retro_input_descriptor *desc = NULL;
|
|
|
|
|
2014-09-02 05:57:53 +02:00
|
|
|
memset(g_extern.system.input_desc_btn, 0,
|
|
|
|
sizeof(g_extern.system.input_desc_btn));
|
|
|
|
|
2015-01-09 21:30:07 +01:00
|
|
|
desc = (const struct retro_input_descriptor*)data;
|
2012-09-09 23:35:23 +02:00
|
|
|
|
|
|
|
for (; desc->description; desc++)
|
|
|
|
{
|
2014-12-05 14:29:40 +01:00
|
|
|
retro_port = desc->port;
|
|
|
|
retro_id = desc->id;
|
2015-01-15 02:48:40 +01:00
|
|
|
|
2015-01-05 01:58:00 +01:00
|
|
|
if (desc->port >= MAX_USERS)
|
2012-09-09 23:35:23 +02:00
|
|
|
continue;
|
|
|
|
|
2014-09-02 05:57:53 +02:00
|
|
|
/* Ignore all others for now. */
|
2014-12-04 14:53:29 +01:00
|
|
|
if (desc->device != RETRO_DEVICE_JOYPAD &&
|
|
|
|
desc->device != RETRO_DEVICE_ANALOG)
|
2012-09-09 23:35:23 +02:00
|
|
|
continue;
|
|
|
|
|
2012-10-01 22:15:48 +02:00
|
|
|
if (desc->id >= RARCH_FIRST_CUSTOM_BIND)
|
2012-09-09 23:35:23 +02:00
|
|
|
continue;
|
|
|
|
|
2014-12-04 14:53:29 +01:00
|
|
|
if (desc->device == RETRO_DEVICE_ANALOG)
|
|
|
|
{
|
2014-12-05 14:29:40 +01:00
|
|
|
switch (retro_id)
|
2014-12-04 14:53:29 +01:00
|
|
|
{
|
|
|
|
case RETRO_DEVICE_ID_ANALOG_X:
|
|
|
|
switch (desc->index)
|
|
|
|
{
|
|
|
|
case RETRO_DEVICE_INDEX_ANALOG_LEFT:
|
2014-12-05 14:29:40 +01:00
|
|
|
g_extern.system.input_desc_btn[retro_port][RARCH_ANALOG_LEFT_X_PLUS] = desc->description;
|
|
|
|
g_extern.system.input_desc_btn[retro_port][RARCH_ANALOG_LEFT_X_MINUS] = desc->description;
|
2014-12-04 14:53:29 +01:00
|
|
|
break;
|
|
|
|
case RETRO_DEVICE_INDEX_ANALOG_RIGHT:
|
2014-12-05 14:29:40 +01:00
|
|
|
g_extern.system.input_desc_btn[retro_port][RARCH_ANALOG_RIGHT_X_PLUS] = desc->description;
|
|
|
|
g_extern.system.input_desc_btn[retro_port][RARCH_ANALOG_RIGHT_X_MINUS] = desc->description;
|
2014-12-04 14:53:29 +01:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case RETRO_DEVICE_ID_ANALOG_Y:
|
|
|
|
switch (desc->index)
|
|
|
|
{
|
|
|
|
case RETRO_DEVICE_INDEX_ANALOG_LEFT:
|
2014-12-05 14:29:40 +01:00
|
|
|
g_extern.system.input_desc_btn[retro_port][RARCH_ANALOG_LEFT_Y_PLUS] = desc->description;
|
|
|
|
g_extern.system.input_desc_btn[retro_port][RARCH_ANALOG_LEFT_Y_MINUS] = desc->description;
|
2014-12-04 14:53:29 +01:00
|
|
|
break;
|
|
|
|
case RETRO_DEVICE_INDEX_ANALOG_RIGHT:
|
2014-12-05 14:29:40 +01:00
|
|
|
g_extern.system.input_desc_btn[retro_port][RARCH_ANALOG_RIGHT_Y_PLUS] = desc->description;
|
|
|
|
g_extern.system.input_desc_btn[retro_port][RARCH_ANALOG_RIGHT_Y_MINUS] = desc->description;
|
2014-12-04 14:53:29 +01:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2014-12-05 14:29:40 +01:00
|
|
|
g_extern.system.input_desc_btn[retro_port][retro_id] = desc->description;
|
2014-12-04 14:53:29 +01:00
|
|
|
}
|
2012-09-09 23:35:23 +02: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",
|
|
|
|
};
|
|
|
|
|
|
|
|
RARCH_LOG("Environ SET_INPUT_DESCRIPTORS:\n");
|
2015-01-05 15:47:27 +01:00
|
|
|
for (p = 0; p < g_settings.input.max_users; p++)
|
2012-09-09 23:35:23 +02:00
|
|
|
{
|
2014-12-05 14:29:40 +01:00
|
|
|
for (retro_id = 0; retro_id < RARCH_FIRST_CUSTOM_BIND; retro_id++)
|
2012-09-09 23:35:23 +02:00
|
|
|
{
|
2014-12-05 14:29:40 +01:00
|
|
|
const char *description = g_extern.system.input_desc_btn[p][retro_id];
|
2015-01-09 21:30:07 +01:00
|
|
|
|
2015-01-15 02:48:40 +01:00
|
|
|
if (!description)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
RARCH_LOG("\tRetroPad, User %u, Button \"%s\" => \"%s\"\n",
|
|
|
|
p + 1, libretro_btn_desc[retro_id], description);
|
2012-09-09 23:35:23 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-12-04 13:45:56 +01:00
|
|
|
g_extern.has_set_input_descriptors = true;
|
|
|
|
|
2012-09-09 23:35:23 +02:00
|
|
|
break;
|
|
|
|
}
|
2014-04-12 13:59:19 +02:00
|
|
|
|
2012-11-25 20:23:31 -05:00
|
|
|
case RETRO_ENVIRONMENT_SET_KEYBOARD_CALLBACK:
|
|
|
|
{
|
2014-09-02 05:57:53 +02:00
|
|
|
const struct retro_keyboard_callback *info =
|
|
|
|
(const struct retro_keyboard_callback*)data;
|
2015-01-09 21:30:07 +01:00
|
|
|
|
|
|
|
RARCH_LOG("Environ SET_KEYBOARD_CALLBACK.\n");
|
2012-11-25 20:23:31 -05:00
|
|
|
g_extern.system.key_event = info->callback;
|
2014-10-30 06:19:51 +01:00
|
|
|
g_extern.frontend_key_event = g_extern.system.key_event;
|
2012-11-25 20:23:31 -05:00
|
|
|
break;
|
|
|
|
}
|
2012-09-09 23:35:23 +02:00
|
|
|
|
2013-02-21 23:44:07 +01:00
|
|
|
case RETRO_ENVIRONMENT_SET_DISK_CONTROL_INTERFACE:
|
|
|
|
RARCH_LOG("Environ SET_DISK_CONTROL_INTERFACE.\n");
|
2014-09-02 05:57:53 +02:00
|
|
|
g_extern.system.disk_control =
|
|
|
|
*(const struct retro_disk_control_callback*)data;
|
2013-02-21 23:44:07 +01:00
|
|
|
break;
|
|
|
|
|
2013-03-27 16:15:15 +01:00
|
|
|
case RETRO_ENVIRONMENT_SET_HW_RENDER:
|
2014-09-02 05:57:53 +02:00
|
|
|
case RETRO_ENVIRONMENT_SET_HW_RENDER | RETRO_ENVIRONMENT_EXPERIMENTAL:
|
2013-03-27 16:15:15 +01:00
|
|
|
{
|
2014-09-02 05:57:53 +02:00
|
|
|
struct retro_hw_render_callback *cb =
|
|
|
|
(struct retro_hw_render_callback*)data;
|
|
|
|
|
2015-01-09 21:30:07 +01:00
|
|
|
RARCH_LOG("Environ SET_HW_RENDER.\n");
|
|
|
|
|
2013-03-29 15:26:47 +01:00
|
|
|
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:
|
2013-12-13 14:19:25 +01:00
|
|
|
#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);
|
2013-03-29 15:26:47 +01:00
|
|
|
break;
|
|
|
|
|
2014-05-03 15:21:14 +02: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 21:02:42 +02:00
|
|
|
#endif
|
2014-05-03 15:21:14 +02:00
|
|
|
|
2013-03-29 15:26:47 +01:00
|
|
|
case RETRO_HW_CONTEXT_OPENGL:
|
2013-06-22 15:06:56 +02:00
|
|
|
case RETRO_HW_CONTEXT_OPENGL_CORE:
|
2013-03-29 15:26:47 +01: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:
|
2013-12-13 14:19:25 +01:00
|
|
|
case RETRO_HW_CONTEXT_OPENGLES3:
|
|
|
|
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);
|
2013-03-29 15:26:47 +01:00
|
|
|
return false;
|
|
|
|
|
2014-05-03 15:21:14 +02:00
|
|
|
case RETRO_HW_CONTEXT_OPENGLES_VERSION:
|
|
|
|
RARCH_ERR("Requesting OpenGLES%u.%u context, but RetroArch is compiled against OpenGL. Cannot use HW context.\n",
|
|
|
|
cb->version_major, cb->version_minor);
|
|
|
|
return false;
|
|
|
|
|
2013-03-29 15:26:47 +01:00
|
|
|
case RETRO_HW_CONTEXT_OPENGL:
|
|
|
|
RARCH_LOG("Requesting OpenGL context.\n");
|
|
|
|
break;
|
2013-06-22 15:06:56 +02:00
|
|
|
|
|
|
|
case RETRO_HW_CONTEXT_OPENGL_CORE:
|
2014-09-02 05:57:53 +02:00
|
|
|
RARCH_LOG("Requesting core OpenGL context (%u.%u).\n",
|
|
|
|
cb->version_major, cb->version_minor);
|
2013-06-22 15:06:56 +02:00
|
|
|
break;
|
2013-03-29 15:26:47 +01:00
|
|
|
#endif
|
|
|
|
|
|
|
|
default:
|
|
|
|
RARCH_LOG("Requesting unknown context.\n");
|
|
|
|
return false;
|
|
|
|
}
|
2015-01-19 04:19:30 +01:00
|
|
|
cb->get_current_framebuffer = video_driver_get_current_framebuffer;
|
|
|
|
cb->get_proc_address = video_driver_get_proc_address;
|
2013-06-22 15:06:56 +02:00
|
|
|
|
2014-09-02 05:57:53 +02:00
|
|
|
if (cmd & RETRO_ENVIRONMENT_EXPERIMENTAL) /* Old ABI. Don't copy garbage. */
|
|
|
|
memcpy(&g_extern.system.hw_render_callback,
|
|
|
|
cb, offsetof(struct retro_hw_render_callback, stencil));
|
2013-06-22 15:06:56 +02:00
|
|
|
else
|
|
|
|
memcpy(&g_extern.system.hw_render_callback, cb, sizeof(*cb));
|
2013-03-27 16:15:15 +01:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2013-05-01 00:56:13 +02:00
|
|
|
case RETRO_ENVIRONMENT_SET_SUPPORT_NO_GAME:
|
|
|
|
{
|
|
|
|
bool state = *(const bool*)data;
|
|
|
|
RARCH_LOG("Environ SET_SUPPORT_NO_GAME: %s.\n", state ? "yes" : "no");
|
2014-08-12 04:57:31 +02:00
|
|
|
g_extern.system.no_content = state;
|
2013-05-01 00:56:13 +02:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2013-07-07 11:15:29 +02:00
|
|
|
case RETRO_ENVIRONMENT_GET_LIBRETRO_PATH:
|
|
|
|
{
|
|
|
|
const char **path = (const char**)data;
|
|
|
|
#ifdef HAVE_DYNAMIC
|
|
|
|
*path = g_settings.libretro;
|
|
|
|
#else
|
2014-04-12 13:59:19 +02:00
|
|
|
*path = NULL;
|
2013-07-07 11:15:29 +02:00
|
|
|
#endif
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2014-09-02 05:57:53 +02:00
|
|
|
/* FIXME - PS3 audio driver needs to be fixed so that threaded
|
|
|
|
* audio works correctly (audio is already on a thread for PS3
|
|
|
|
* audio driver so that's probably the problem) */
|
2014-01-22 02:20:03 +01:00
|
|
|
#if defined(HAVE_THREADS) && !defined(__CELLOS_LV2__)
|
2013-07-14 13:09:53 +02:00
|
|
|
case RETRO_ENVIRONMENT_SET_AUDIO_CALLBACK:
|
|
|
|
{
|
|
|
|
RARCH_LOG("Environ SET_AUDIO_CALLBACK.\n");
|
2014-09-02 05:57:53 +02:00
|
|
|
const struct retro_audio_callback *info =
|
|
|
|
(const struct retro_audio_callback*)data;
|
2013-07-14 13:09:53 +02:00
|
|
|
|
2014-09-30 21:23:00 +02:00
|
|
|
if (driver.recording_data) // A/V sync is a must.
|
2013-07-14 13:09:53 +02:00
|
|
|
return false;
|
2013-07-15 20:36:18 +02:00
|
|
|
|
|
|
|
#ifdef HAVE_NETPLAY
|
|
|
|
if (g_extern.netplay_enable)
|
|
|
|
return false;
|
|
|
|
#endif
|
|
|
|
|
2013-09-14 17:26:10 +02:00
|
|
|
g_extern.system.audio_callback = *info;
|
2013-07-14 13:09:53 +02:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
case RETRO_ENVIRONMENT_SET_FRAME_TIME_CALLBACK:
|
2013-07-14 13:43:01 +02:00
|
|
|
{
|
2015-01-09 21:30:07 +01:00
|
|
|
const struct retro_frame_time_callback *info =
|
|
|
|
(const struct retro_frame_time_callback*)data;
|
|
|
|
|
2013-07-14 13:09:53 +02:00
|
|
|
RARCH_LOG("Environ SET_FRAME_TIME_CALLBACK.\n");
|
2013-07-15 20:36:18 +02:00
|
|
|
|
|
|
|
#ifdef HAVE_NETPLAY
|
2014-09-02 05:57:53 +02:00
|
|
|
/* retro_run() will be called in very strange and
|
|
|
|
* mysterious ways, have to disable it. */
|
|
|
|
if (g_extern.netplay_enable)
|
2013-07-14 13:09:53 +02:00
|
|
|
return false;
|
2013-07-15 20:36:18 +02:00
|
|
|
#endif
|
|
|
|
|
2013-07-14 13:43:01 +02:00
|
|
|
g_extern.system.frame_time = *info;
|
2013-07-14 13:09:53 +02:00
|
|
|
break;
|
2013-07-14 13:43:01 +02:00
|
|
|
}
|
2013-08-25 01:37:15 +02:00
|
|
|
|
2013-09-25 23:21:32 +02:00
|
|
|
case RETRO_ENVIRONMENT_GET_RUMBLE_INTERFACE:
|
|
|
|
{
|
2014-09-02 05:57:53 +02:00
|
|
|
struct retro_rumble_interface *iface =
|
|
|
|
(struct retro_rumble_interface*)data;
|
2015-01-09 21:30:07 +01:00
|
|
|
|
|
|
|
RARCH_LOG("Environ GET_RUMBLE_INTERFACE.\n");
|
2015-01-19 04:16:43 +01:00
|
|
|
iface->set_rumble_state = input_driver_set_rumble_state;
|
2013-09-25 23:21:32 +02:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2013-11-02 21:39:43 +01:00
|
|
|
case RETRO_ENVIRONMENT_GET_INPUT_DEVICE_CAPABILITIES:
|
|
|
|
{
|
|
|
|
uint64_t *mask = (uint64_t*)data;
|
2015-01-09 21:30:07 +01:00
|
|
|
|
|
|
|
RARCH_LOG("Environ GET_INPUT_DEVICE_CAPABILITIES.\n");
|
2014-09-02 05:57:53 +02:00
|
|
|
if (driver.input &&
|
|
|
|
driver.input->get_capabilities && driver.input_data)
|
2013-11-02 21:39:43 +01:00
|
|
|
*mask = driver.input->get_capabilities(driver.input_data);
|
|
|
|
else
|
|
|
|
return false;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2013-11-03 12:06:14 +01:00
|
|
|
case RETRO_ENVIRONMENT_GET_SENSOR_INTERFACE:
|
|
|
|
{
|
2014-09-02 05:57:53 +02:00
|
|
|
struct retro_sensor_interface *iface =
|
|
|
|
(struct retro_sensor_interface*)data;
|
2015-01-09 21:30:07 +01:00
|
|
|
|
|
|
|
RARCH_LOG("Environ GET_SENSOR_INTERFACE.\n");
|
2015-01-18 19:28:51 +01:00
|
|
|
iface->set_sensor_state = input_sensor_set_state;
|
|
|
|
iface->get_sensor_input = input_sensor_get_input;
|
2013-11-03 12:06:14 +01:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2013-11-12 16:00:18 +01:00
|
|
|
case RETRO_ENVIRONMENT_GET_CAMERA_INTERFACE:
|
2013-11-12 12:00:57 +01:00
|
|
|
{
|
2014-09-02 05:57:53 +02:00
|
|
|
struct retro_camera_callback *cb =
|
|
|
|
(struct retro_camera_callback*)data;
|
2015-01-09 21:30:07 +01:00
|
|
|
|
|
|
|
RARCH_LOG("Environ GET_CAMERA_INTERFACE.\n");
|
2013-11-12 16:00:18 +01:00
|
|
|
cb->start = driver_camera_start;
|
|
|
|
cb->stop = driver_camera_stop;
|
|
|
|
g_extern.system.camera_callback = *cb;
|
2014-10-01 22:34:48 +02:00
|
|
|
driver.camera_active = cb->caps != 0;
|
2013-11-12 14:14:27 +01:00
|
|
|
break;
|
2013-11-12 12:00:57 +01:00
|
|
|
}
|
2014-08-14 02:55:04 +02:00
|
|
|
|
2013-12-19 01:51:51 +01:00
|
|
|
case RETRO_ENVIRONMENT_GET_LOCATION_INTERFACE:
|
|
|
|
{
|
2014-09-02 05:57:53 +02:00
|
|
|
struct retro_location_callback *cb =
|
|
|
|
(struct retro_location_callback*)data;
|
2015-01-09 21:30:07 +01:00
|
|
|
|
|
|
|
RARCH_LOG("Environ GET_LOCATION_INTERFACE.\n");
|
2013-12-19 05:51:25 +01:00
|
|
|
cb->start = driver_location_start;
|
|
|
|
cb->stop = driver_location_stop;
|
2013-12-19 15:47:47 +01:00
|
|
|
cb->get_position = driver_location_get_position;
|
2013-12-19 05:51:25 +01:00
|
|
|
cb->set_interval = driver_location_set_interval;
|
2013-12-19 02:54:46 +01:00
|
|
|
g_extern.system.location_callback = *cb;
|
2014-10-01 22:34:48 +02:00
|
|
|
driver.location_active = true;
|
2013-12-19 01:51:51 +01:00
|
|
|
break;
|
|
|
|
}
|
2013-11-12 12:00:57 +01:00
|
|
|
|
2013-12-09 10:32:12 +01:00
|
|
|
case RETRO_ENVIRONMENT_GET_LOG_INTERFACE:
|
|
|
|
{
|
|
|
|
struct retro_log_callback *cb = (struct retro_log_callback*)data;
|
2015-01-09 21:30:07 +01:00
|
|
|
|
|
|
|
RARCH_LOG("Environ GET_LOG_INTERFACE.\n");
|
2013-12-09 10:32:12 +01:00
|
|
|
cb->log = rarch_log_libretro;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2013-12-17 19:10:21 +01:00
|
|
|
case RETRO_ENVIRONMENT_GET_PERF_INTERFACE:
|
|
|
|
{
|
|
|
|
struct retro_perf_callback *cb = (struct retro_perf_callback*)data;
|
2015-01-09 21:30:07 +01:00
|
|
|
|
|
|
|
RARCH_LOG("Environ GET_PERF_INTERFACE.\n");
|
2013-12-18 19:10:57 +01:00
|
|
|
cb->get_time_usec = rarch_get_time_usec;
|
|
|
|
cb->get_cpu_features = rarch_get_cpu_features;
|
|
|
|
cb->get_perf_counter = rarch_get_perf_counter;
|
2014-09-02 05:57:53 +02:00
|
|
|
cb->perf_register = retro_perf_register; /* libretro specific path. */
|
2013-12-18 19:10:57 +01:00
|
|
|
cb->perf_start = rarch_perf_start;
|
|
|
|
cb->perf_stop = rarch_perf_stop;
|
2014-09-02 05:57:53 +02:00
|
|
|
cb->perf_log = retro_perf_log; /* libretro specific path. */
|
2013-12-17 19:10:21 +01:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2014-02-20 22:48:31 +01:00
|
|
|
case RETRO_ENVIRONMENT_GET_CONTENT_DIRECTORY:
|
|
|
|
{
|
|
|
|
const char **dir = (const char**)data;
|
2015-01-09 21:30:07 +01:00
|
|
|
|
2014-09-02 05:57:53 +02:00
|
|
|
*dir = *g_settings.content_directory ?
|
|
|
|
g_settings.content_directory : NULL;
|
|
|
|
RARCH_LOG("Environ CONTENT_DIRECTORY: \"%s\".\n",
|
|
|
|
g_settings.content_directory);
|
2014-02-20 22:48:31 +01:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
case RETRO_ENVIRONMENT_SET_SYSTEM_AV_INFO:
|
|
|
|
{
|
2014-02-20 23:05:56 +01:00
|
|
|
RARCH_LOG("Environ SET_SYSTEM_AV_INFO.\n");
|
2014-09-02 05:57:53 +02:00
|
|
|
return driver_update_system_av_info(
|
|
|
|
(const struct retro_system_av_info*)data);
|
2014-02-20 22:48:31 +01:00
|
|
|
}
|
|
|
|
|
2014-04-04 17:04:01 +02:00
|
|
|
case RETRO_ENVIRONMENT_SET_SUBSYSTEM_INFO:
|
2014-04-04 14:58:42 +02:00
|
|
|
{
|
2014-04-04 16:02:53 +02:00
|
|
|
unsigned i, j;
|
2014-09-02 05:57:53 +02:00
|
|
|
const struct retro_subsystem_info *info =
|
|
|
|
(const struct retro_subsystem_info*)data;
|
|
|
|
|
2015-01-09 21:30:07 +01:00
|
|
|
RARCH_LOG("Environ SET_SUBSYSTEM_INFO.\n");
|
|
|
|
|
2014-04-04 14:58:42 +02:00
|
|
|
for (i = 0; info[i].ident; i++)
|
2014-04-04 16:02:53 +02: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 20:01:27 +02:00
|
|
|
RARCH_LOG(" Content:\n");
|
2014-04-04 16:02:53 +02:00
|
|
|
for (j = 0; j < info[i].num_roms; j++)
|
|
|
|
{
|
|
|
|
RARCH_LOG(" %s (%s)\n",
|
2014-09-02 05:57:53 +02:00
|
|
|
info[i].roms[j].desc, info[i].roms[j].required ?
|
|
|
|
"required" : "optional");
|
2014-04-04 16:02:53 +02:00
|
|
|
}
|
|
|
|
}
|
2014-04-04 14:58:42 +02:00
|
|
|
|
|
|
|
free(g_extern.system.special);
|
2014-09-02 05:57:53 +02:00
|
|
|
g_extern.system.special = (struct retro_subsystem_info*)
|
|
|
|
calloc(i, sizeof(*g_extern.system.special));
|
|
|
|
|
2014-04-04 14:58:42 +02:00
|
|
|
if (!g_extern.system.special)
|
|
|
|
return false;
|
|
|
|
|
2014-09-02 05:57:53 +02:00
|
|
|
memcpy(g_extern.system.special, info,
|
|
|
|
i * sizeof(*g_extern.system.special));
|
2014-04-04 17:04:01 +02:00
|
|
|
g_extern.system.num_special = i;
|
2014-04-04 14:58:42 +02:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2014-04-12 13:22:24 +02:00
|
|
|
case RETRO_ENVIRONMENT_SET_CONTROLLER_INFO:
|
|
|
|
{
|
|
|
|
unsigned i, j;
|
2014-09-02 05:57:53 +02:00
|
|
|
const struct retro_controller_info *info =
|
|
|
|
(const struct retro_controller_info*)data;
|
|
|
|
|
2015-01-09 21:30:07 +01:00
|
|
|
RARCH_LOG("Environ SET_CONTROLLER_INFO.\n");
|
|
|
|
|
2014-04-12 13:22:24 +02:00
|
|
|
for (i = 0; info[i].types; i++)
|
|
|
|
{
|
2014-04-12 19:06:37 +02:00
|
|
|
RARCH_LOG("Controller port: %u\n", i + 1);
|
2014-04-12 13:22:24 +02:00
|
|
|
for (j = 0; j < info[i].num_types; j++)
|
2014-09-02 05:57:53 +02:00
|
|
|
RARCH_LOG(" %s (ID: %u)\n", info[i].types[j].desc,
|
|
|
|
info[i].types[j].id);
|
2014-04-12 13:22:24 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
free(g_extern.system.ports);
|
2014-09-02 05:57:53 +02:00
|
|
|
g_extern.system.ports = (struct retro_controller_info*)
|
|
|
|
calloc(i, sizeof(*g_extern.system.ports));
|
2014-04-12 13:22:24 +02:00
|
|
|
if (!g_extern.system.ports)
|
|
|
|
return false;
|
|
|
|
|
2014-09-02 05:57:53 +02:00
|
|
|
memcpy(g_extern.system.ports, info,
|
|
|
|
i * sizeof(*g_extern.system.ports));
|
2014-04-12 13:22:24 +02:00
|
|
|
g_extern.system.num_ports = i;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2014-06-07 13:19:28 +02:00
|
|
|
case RETRO_ENVIRONMENT_SET_GEOMETRY:
|
|
|
|
{
|
2014-09-02 05:57:53 +02:00
|
|
|
const struct retro_game_geometry *in_geom =
|
|
|
|
(const struct retro_game_geometry*)data;
|
2014-06-07 13:19:28 +02:00
|
|
|
struct retro_game_geometry *geom = &g_extern.system.av_info.geometry;
|
|
|
|
|
2015-01-09 21:30:07 +01:00
|
|
|
RARCH_LOG("Environ SET_GEOMETRY.\n");
|
|
|
|
|
2014-09-02 05:57:53 +02:00
|
|
|
/* Can potentially be called every frame,
|
|
|
|
* don't do anything unless required. */
|
2014-06-07 13:19:28 +02: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;
|
|
|
|
RARCH_LOG("SET_GEOMETRY: %ux%u, aspect: %.3f.\n",
|
|
|
|
geom->base_width, geom->base_height, geom->aspect_ratio);
|
|
|
|
|
2014-09-02 05:57:53 +02:00
|
|
|
/* Forces recomputation of aspect ratios if
|
|
|
|
* using core-dependent aspect ratios. */
|
2014-08-16 18:46:43 +02:00
|
|
|
rarch_main_command(RARCH_CMD_VIDEO_SET_ASPECT_RATIO);
|
2014-06-07 13:19:28 +02:00
|
|
|
|
2014-09-02 05:57:53 +02:00
|
|
|
/* TODO: Figure out what to do, if anything, with recording. */
|
2014-06-07 13:19:28 +02:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2014-09-02 05:57:53 +02:00
|
|
|
/* Private extensions for internal use, not part of libretro API. */
|
2014-04-12 13:22:24 +02:00
|
|
|
case RETRO_ENVIRONMENT_SET_LIBRETRO_PATH:
|
|
|
|
RARCH_LOG("Environ (Private) SET_LIBRETRO_PATH.\n");
|
|
|
|
|
|
|
|
if (path_file_exists((const char*)data))
|
2014-09-02 05:57:53 +02:00
|
|
|
strlcpy(g_settings.libretro, (const char*)data,
|
|
|
|
sizeof(g_settings.libretro));
|
2014-04-12 13:22:24 +02:00
|
|
|
else
|
|
|
|
return false;
|
|
|
|
break;
|
|
|
|
|
2013-08-25 01:37:15 +02:00
|
|
|
case RETRO_ENVIRONMENT_EXEC:
|
2013-08-25 19:42:24 +02:00
|
|
|
case RETRO_ENVIRONMENT_EXEC_ESCAPE:
|
2013-08-25 01:37:15 +02:00
|
|
|
|
2013-08-25 11:07:06 +02:00
|
|
|
if (data)
|
2014-09-02 05:57:53 +02:00
|
|
|
strlcpy(g_extern.fullpath, (const char*)data,
|
|
|
|
sizeof(g_extern.fullpath));
|
2013-08-25 01:37:15 +02:00
|
|
|
else
|
|
|
|
*g_extern.fullpath = '\0';
|
|
|
|
|
2013-11-07 21:44:44 +01:00
|
|
|
#if defined(RARCH_CONSOLE)
|
2014-10-02 21:39:29 +02:00
|
|
|
if (driver.frontend_ctx && driver.frontend_ctx->set_fork)
|
|
|
|
driver.frontend_ctx->set_fork(true, true);
|
2013-08-25 01:37:15 +02:00
|
|
|
#elif defined(HAVE_DYNAMIC)
|
2014-09-03 18:36:46 +02:00
|
|
|
rarch_main_set_state(RARCH_ACTION_STATE_LOAD_CONTENT);
|
2013-08-25 01:37:15 +02:00
|
|
|
#endif
|
2013-08-25 19:42:24 +02:00
|
|
|
|
|
|
|
if (cmd == RETRO_ENVIRONMENT_EXEC_ESCAPE)
|
|
|
|
{
|
|
|
|
RARCH_LOG("Environ (Private) EXEC_ESCAPE.\n");
|
|
|
|
g_extern.exec = true;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
RARCH_LOG("Environ (Private) EXEC.\n");
|
2013-08-25 01:37:15 +02:00
|
|
|
|
|
|
|
break;
|
2013-12-26 02:03:11 +01:00
|
|
|
|
2011-10-27 23:40:34 +02:00
|
|
|
default:
|
2012-04-21 23:25:32 +02:00
|
|
|
RARCH_LOG("Environ UNSUPPORTED (#%u).\n", cmd);
|
2011-10-27 23:40:34 +02:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|