2014-09-23 03:03:56 +02:00
|
|
|
/* RetroArch - A frontend for libretro.
|
|
|
|
* Copyright (C) 2010-2014 - Hans-Kristian Arntzen
|
2016-01-10 04:06:50 +01:00
|
|
|
* Copyright (C) 2011-2016 - Daniel De Matteis
|
2015-01-07 17:46:50 +01:00
|
|
|
* Copyright (C) 2012-2015 - Michael Lelli
|
2014-09-23 03:03:56 +02:00
|
|
|
*
|
|
|
|
* RetroArch is free software: you can redistribute it and/or modify it under the terms
|
|
|
|
* of the GNU General Public License as published by the Free Software Found-
|
|
|
|
* ation, either version 3 of the License, or (at your option) any later version.
|
|
|
|
*
|
|
|
|
* RetroArch is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY;
|
|
|
|
* without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR
|
|
|
|
* PURPOSE. See the GNU General Public License for more details.
|
|
|
|
*
|
|
|
|
* You should have received a copy of the GNU General Public License along with RetroArch.
|
|
|
|
* If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <stdio.h>
|
|
|
|
#include <stdlib.h>
|
|
|
|
#include <string.h>
|
|
|
|
#include <ctype.h>
|
|
|
|
#include <errno.h>
|
2015-06-30 00:38:10 +02:00
|
|
|
|
|
|
|
#include <retro_inline.h>
|
|
|
|
#include <boolean.h>
|
2016-03-20 14:53:54 +01:00
|
|
|
#include <lists/string_list.h>
|
2016-05-10 19:03:53 +02:00
|
|
|
#include <libretro.h>
|
2015-06-30 00:38:10 +02:00
|
|
|
|
2016-09-01 05:36:52 +02:00
|
|
|
#include "configuration.h"
|
2016-03-22 02:56:06 +01:00
|
|
|
#include "dynamic.h"
|
2016-05-08 05:29:10 +02:00
|
|
|
#include "core.h"
|
2016-01-27 08:13:54 +01:00
|
|
|
#include "msg_hash.h"
|
2016-05-09 18:20:52 +02:00
|
|
|
#include "managers/state_manager.h"
|
2016-09-01 05:36:52 +02:00
|
|
|
#include "runloop.h"
|
2016-01-27 08:58:31 +01:00
|
|
|
#include "system.h"
|
2016-09-01 05:36:52 +02:00
|
|
|
#include "verbosity.h"
|
2015-11-24 01:26:59 +01:00
|
|
|
#include "gfx/video_driver.h"
|
2015-11-23 19:40:09 +01:00
|
|
|
#include "audio/audio_driver.h"
|
2014-09-23 03:03:56 +02:00
|
|
|
|
2014-10-20 21:32:53 +02:00
|
|
|
#ifdef HAVE_NETPLAY
|
2016-05-19 11:46:54 +02:00
|
|
|
#include "network/netplay/netplay.h"
|
2014-10-20 21:32:53 +02:00
|
|
|
#endif
|
|
|
|
|
2016-01-28 04:41:54 +01:00
|
|
|
static struct retro_core_t core;
|
2016-09-06 03:52:29 +02:00
|
|
|
static bool core_inited;
|
|
|
|
static bool core_symbols_inited;
|
2016-09-03 17:50:04 +01:00
|
|
|
static bool core_game_loaded;
|
2016-01-28 06:21:29 +01:00
|
|
|
static unsigned core_poll_type;
|
|
|
|
static bool core_input_polled;
|
2016-05-08 00:31:21 +02:00
|
|
|
static bool core_has_set_input_descriptors = false;
|
|
|
|
static struct retro_callbacks retro_ctx;
|
2016-01-27 01:16:53 +01:00
|
|
|
|
2016-05-08 00:31:21 +02:00
|
|
|
static void core_input_state_poll_maybe(void)
|
|
|
|
{
|
|
|
|
if (core_poll_type == POLL_TYPE_NORMAL)
|
|
|
|
input_poll();
|
|
|
|
}
|
|
|
|
|
|
|
|
static int16_t core_input_state_poll(unsigned port,
|
2016-01-27 01:16:53 +01:00
|
|
|
unsigned device, unsigned idx, unsigned id)
|
|
|
|
{
|
2016-01-28 06:20:44 +01:00
|
|
|
if (core_poll_type == POLL_TYPE_LATE)
|
2016-01-27 01:16:53 +01:00
|
|
|
{
|
2016-01-28 06:21:29 +01:00
|
|
|
if (!core_input_polled)
|
2016-01-27 01:16:53 +01:00
|
|
|
input_poll();
|
|
|
|
|
2016-01-28 06:21:29 +01:00
|
|
|
core_input_polled = true;
|
2016-01-27 01:16:53 +01:00
|
|
|
}
|
|
|
|
return input_state(port, device, idx, id);
|
|
|
|
}
|
|
|
|
|
2016-05-08 01:33:57 +02:00
|
|
|
void core_set_input_state(retro_ctx_input_state_info_t *info)
|
2015-12-05 16:54:03 +01:00
|
|
|
{
|
2016-05-08 01:33:57 +02:00
|
|
|
core.retro_set_input_state(info->cb);
|
2015-12-05 16:54:03 +01:00
|
|
|
}
|
|
|
|
|
2015-01-08 18:39:17 +01:00
|
|
|
/**
|
2016-05-08 00:31:21 +02:00
|
|
|
* core_init_libretro_cbs:
|
2015-01-08 18:39:17 +01:00
|
|
|
* @data : pointer to retro_callbacks object
|
|
|
|
*
|
|
|
|
* Initializes libretro callbacks, and binds the libretro callbacks
|
|
|
|
* to default callback functions.
|
|
|
|
**/
|
2016-05-08 00:31:21 +02:00
|
|
|
static bool core_init_libretro_cbs(void *data)
|
2014-09-23 03:03:56 +02:00
|
|
|
{
|
2014-09-23 04:42:49 +02:00
|
|
|
struct retro_callbacks *cbs = (struct retro_callbacks*)data;
|
2016-01-24 01:59:43 +01:00
|
|
|
#ifdef HAVE_NETPLAY
|
2015-07-08 17:22:35 +02:00
|
|
|
global_t *global = global_get_ptr();
|
2016-01-24 01:59:43 +01:00
|
|
|
#endif
|
2014-09-23 04:42:49 +02:00
|
|
|
|
|
|
|
if (!cbs)
|
2016-01-27 03:14:36 +01:00
|
|
|
return false;
|
2014-09-23 04:42:49 +02:00
|
|
|
|
2015-12-04 00:26:09 +01:00
|
|
|
core.retro_set_video_refresh(video_driver_frame);
|
2015-10-06 14:34:09 -03:00
|
|
|
core.retro_set_audio_sample(audio_driver_sample);
|
|
|
|
core.retro_set_audio_sample_batch(audio_driver_sample_batch);
|
2016-05-08 00:31:21 +02:00
|
|
|
core.retro_set_input_state(core_input_state_poll);
|
|
|
|
core.retro_set_input_poll(core_input_state_poll_maybe);
|
2014-09-23 03:03:56 +02:00
|
|
|
|
2016-05-08 01:33:57 +02:00
|
|
|
core_set_default_callbacks(cbs);
|
2014-09-23 03:55:19 +02:00
|
|
|
|
2014-09-23 03:03:56 +02:00
|
|
|
#ifdef HAVE_NETPLAY
|
2015-12-05 16:38:24 +01:00
|
|
|
if (!netplay_driver_ctl(RARCH_NETPLAY_CTL_IS_DATA_INITED, NULL))
|
2016-01-27 03:14:36 +01:00
|
|
|
return true;
|
2014-09-23 03:03:56 +02:00
|
|
|
|
2016-01-27 01:16:53 +01:00
|
|
|
/* Force normal poll type for netplay. */
|
2016-01-28 06:20:44 +01:00
|
|
|
core_poll_type = POLL_TYPE_NORMAL;
|
2016-01-27 01:16:53 +01:00
|
|
|
|
2015-07-27 17:18:10 +02:00
|
|
|
if (global->netplay.is_spectate)
|
2014-09-23 03:03:56 +02:00
|
|
|
{
|
2015-10-06 14:34:09 -03:00
|
|
|
core.retro_set_input_state(
|
2015-07-27 17:18:10 +02:00
|
|
|
(global->netplay.is_client ?
|
2014-09-23 03:03:56 +02:00
|
|
|
input_state_spectate_client : input_state_spectate)
|
|
|
|
);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2015-10-06 14:34:09 -03:00
|
|
|
core.retro_set_video_refresh(video_frame_net);
|
|
|
|
core.retro_set_audio_sample(audio_sample_net);
|
|
|
|
core.retro_set_audio_sample_batch(audio_sample_batch_net);
|
|
|
|
core.retro_set_input_state(input_state_net);
|
2014-09-23 03:03:56 +02:00
|
|
|
}
|
|
|
|
#endif
|
2016-01-27 03:14:36 +01:00
|
|
|
|
|
|
|
return true;
|
2014-09-23 03:03:56 +02:00
|
|
|
}
|
|
|
|
|
2016-05-08 01:33:57 +02:00
|
|
|
/**
|
|
|
|
* core_set_default_callbacks:
|
|
|
|
* @data : pointer to retro_callbacks object
|
|
|
|
*
|
|
|
|
* Binds the libretro callbacks to default callback functions.
|
|
|
|
**/
|
|
|
|
bool core_set_default_callbacks(void *data)
|
|
|
|
{
|
|
|
|
struct retro_callbacks *cbs = (struct retro_callbacks*)data;
|
|
|
|
|
|
|
|
if (!cbs)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
cbs->frame_cb = video_driver_frame;
|
|
|
|
cbs->sample_cb = audio_driver_sample;
|
|
|
|
cbs->sample_batch_cb = audio_driver_sample_batch;
|
|
|
|
cbs->state_cb = core_input_state_poll;
|
|
|
|
cbs->poll_cb = input_poll;
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
bool core_deinit(void *data)
|
|
|
|
{
|
|
|
|
struct retro_callbacks *cbs = (struct retro_callbacks*)data;
|
|
|
|
|
|
|
|
if (!cbs)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
cbs->frame_cb = NULL;
|
|
|
|
cbs->sample_cb = NULL;
|
|
|
|
cbs->sample_batch_cb = NULL;
|
|
|
|
cbs->state_cb = NULL;
|
|
|
|
cbs->poll_cb = NULL;
|
|
|
|
|
2016-09-06 03:52:29 +02:00
|
|
|
core_inited = false;
|
|
|
|
|
2016-05-08 01:33:57 +02:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool core_uninit_libretro_callbacks(void)
|
|
|
|
{
|
|
|
|
return core_deinit(&retro_ctx);
|
|
|
|
}
|
|
|
|
|
2015-01-08 18:39:17 +01:00
|
|
|
/**
|
2016-05-08 00:31:21 +02:00
|
|
|
* core_set_rewind_callbacks:
|
2015-01-08 18:39:17 +01:00
|
|
|
*
|
|
|
|
* Sets the audio sampling callbacks based on whether or not
|
|
|
|
* rewinding is currently activated.
|
|
|
|
**/
|
2016-05-08 01:33:57 +02:00
|
|
|
bool core_set_rewind_callbacks(void)
|
2014-09-23 03:03:56 +02:00
|
|
|
{
|
2015-09-16 11:14:43 +02:00
|
|
|
if (state_manager_frame_is_reversed())
|
2014-09-23 03:03:56 +02:00
|
|
|
{
|
2015-10-06 14:34:09 -03:00
|
|
|
core.retro_set_audio_sample(audio_driver_sample_rewind);
|
|
|
|
core.retro_set_audio_sample_batch(audio_driver_sample_batch_rewind);
|
2014-09-23 03:03:56 +02:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2015-10-06 14:34:09 -03:00
|
|
|
core.retro_set_audio_sample(audio_driver_sample);
|
|
|
|
core.retro_set_audio_sample_batch(audio_driver_sample_batch);
|
2014-09-23 03:03:56 +02:00
|
|
|
}
|
2016-05-08 01:33:57 +02:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool core_set_cheat(retro_ctx_cheat_info_t *info)
|
|
|
|
{
|
|
|
|
core.retro_cheat_set(info->index, info->enabled, info->code);
|
|
|
|
return true;
|
2014-09-23 03:03:56 +02:00
|
|
|
}
|
2016-01-27 03:08:49 +01:00
|
|
|
|
2016-05-08 01:33:57 +02:00
|
|
|
bool core_reset_cheat(void)
|
2016-01-27 03:08:49 +01:00
|
|
|
{
|
2016-05-08 01:33:57 +02:00
|
|
|
core.retro_cheat_reset();
|
|
|
|
return true;
|
|
|
|
}
|
2016-01-27 04:54:22 +01:00
|
|
|
|
2016-05-08 01:33:57 +02:00
|
|
|
bool core_api_version(retro_ctx_api_info_t *api)
|
|
|
|
{
|
|
|
|
if (!api)
|
|
|
|
return false;
|
|
|
|
api->version = core.retro_api_version();
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool core_set_poll_type(unsigned *type)
|
|
|
|
{
|
|
|
|
core_poll_type = *type;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2016-07-23 07:06:32 +02:00
|
|
|
void core_uninit_symbols(void)
|
|
|
|
{
|
|
|
|
uninit_libretro_sym(&core);
|
2016-09-06 03:52:29 +02:00
|
|
|
core_symbols_inited = false;
|
2016-07-23 07:06:32 +02:00
|
|
|
}
|
|
|
|
|
2016-05-08 01:33:57 +02:00
|
|
|
bool core_init_symbols(enum rarch_core_type *type)
|
|
|
|
{
|
|
|
|
if (!type)
|
|
|
|
return false;
|
|
|
|
init_libretro_sym(*type, &core);
|
2016-09-06 03:52:29 +02:00
|
|
|
core_symbols_inited = true;
|
2016-05-08 01:33:57 +02:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool core_set_controller_port_device(retro_ctx_controller_info_t *pad)
|
|
|
|
{
|
|
|
|
if (!pad)
|
|
|
|
return false;
|
|
|
|
core.retro_set_controller_port_device(pad->port, pad->device);
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool core_get_memory(retro_ctx_memory_info_t *info)
|
|
|
|
{
|
|
|
|
if (!info)
|
|
|
|
return false;
|
|
|
|
info->size = core.retro_get_memory_size(info->id);
|
|
|
|
info->data = core.retro_get_memory_data(info->id);
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool core_load_game(retro_ctx_load_content_info_t *load_info)
|
|
|
|
{
|
|
|
|
if (!load_info)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
if (load_info->special)
|
2016-09-03 17:50:04 +01:00
|
|
|
core_game_loaded = core.retro_load_game_special(load_info->special->id, load_info->info, load_info->content->size);
|
|
|
|
else
|
|
|
|
core_game_loaded = core.retro_load_game(*load_info->content->elems[0].data ? load_info->info : NULL);
|
|
|
|
|
|
|
|
return core_game_loaded;
|
2016-05-08 01:33:57 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
bool core_get_system_info(struct retro_system_info *system)
|
|
|
|
{
|
|
|
|
if (!system)
|
|
|
|
return false;
|
|
|
|
core.retro_get_system_info(system);
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool core_unserialize(retro_ctx_serialize_info_t *info)
|
|
|
|
{
|
|
|
|
if (!info)
|
|
|
|
return false;
|
|
|
|
if (!core.retro_unserialize(info->data_const, info->size))
|
|
|
|
return false;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool core_serialize(retro_ctx_serialize_info_t *info)
|
|
|
|
{
|
|
|
|
if (!info)
|
|
|
|
return false;
|
|
|
|
if (!core.retro_serialize(info->data, info->size))
|
|
|
|
return false;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool core_serialize_size(retro_ctx_size_info_t *info)
|
|
|
|
{
|
|
|
|
if (!info)
|
|
|
|
return false;
|
|
|
|
info->size = core.retro_serialize_size();
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool core_frame(retro_ctx_frame_info_t *info)
|
|
|
|
{
|
|
|
|
if (!info || !retro_ctx.frame_cb)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
retro_ctx.frame_cb(
|
|
|
|
info->data, info->width, info->height, info->pitch);
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool core_poll(void)
|
|
|
|
{
|
|
|
|
if (!retro_ctx.poll_cb)
|
|
|
|
return false;
|
|
|
|
retro_ctx.poll_cb();
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool core_set_environment(retro_ctx_environ_info_t *info)
|
|
|
|
{
|
|
|
|
if (!info)
|
|
|
|
return false;
|
|
|
|
core.retro_set_environment(info->env);
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool core_get_system_av_info(struct retro_system_av_info *av_info)
|
|
|
|
{
|
|
|
|
if (!av_info)
|
|
|
|
return false;
|
|
|
|
core.retro_get_system_av_info(av_info);
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool core_reset(void)
|
|
|
|
{
|
|
|
|
core.retro_reset();
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool core_init(void)
|
|
|
|
{
|
|
|
|
core.retro_init();
|
2016-09-06 03:52:29 +02:00
|
|
|
core_inited = true;
|
2016-05-08 01:33:57 +02:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool core_unload(void)
|
|
|
|
{
|
|
|
|
core.retro_deinit();
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2016-07-23 07:06:32 +02:00
|
|
|
|
2016-05-08 01:33:57 +02:00
|
|
|
bool core_unload_game(void)
|
|
|
|
{
|
2016-05-08 14:00:51 +02:00
|
|
|
video_driver_deinit_hw_context();
|
2016-05-08 18:00:32 +02:00
|
|
|
audio_driver_stop();
|
2016-05-08 01:33:57 +02:00
|
|
|
core.retro_unload_game();
|
2016-09-03 17:50:04 +01:00
|
|
|
core_game_loaded = false;
|
2016-05-08 01:33:57 +02:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool core_run(void)
|
|
|
|
{
|
|
|
|
switch (core_poll_type)
|
2016-01-27 03:08:49 +01:00
|
|
|
{
|
2016-05-08 01:33:57 +02:00
|
|
|
case POLL_TYPE_EARLY:
|
|
|
|
input_poll();
|
2016-04-06 00:20:14 +02:00
|
|
|
break;
|
2016-05-08 01:33:57 +02:00
|
|
|
case POLL_TYPE_LATE:
|
|
|
|
core_input_polled = false;
|
2016-01-27 03:08:49 +01:00
|
|
|
break;
|
|
|
|
}
|
2016-05-08 01:33:57 +02:00
|
|
|
if (core.retro_run)
|
|
|
|
core.retro_run();
|
|
|
|
if (core_poll_type == POLL_TYPE_LATE && !core_input_polled)
|
|
|
|
input_poll();
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool core_load(void)
|
|
|
|
{
|
|
|
|
settings_t *settings = config_get_ptr();
|
|
|
|
core_poll_type = settings->input.poll_type_behavior;
|
|
|
|
|
|
|
|
if (!core_verify_api_version())
|
|
|
|
return false;
|
|
|
|
if (!core_init_libretro_cbs(&retro_ctx))
|
|
|
|
return false;
|
|
|
|
|
|
|
|
core_get_system_av_info(video_viewport_get_system_av_info());
|
|
|
|
runloop_ctl(RUNLOOP_CTL_SET_FRAME_LIMIT, NULL);
|
2016-01-27 03:08:49 +01:00
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
2016-05-08 01:33:57 +02:00
|
|
|
|
|
|
|
bool core_verify_api_version(void)
|
|
|
|
{
|
|
|
|
unsigned api_version = core.retro_api_version();
|
2016-06-20 04:23:00 +02:00
|
|
|
RARCH_LOG("%s: %u\n",
|
|
|
|
msg_hash_to_str(MSG_VERSION_OF_LIBRETRO_API),
|
|
|
|
api_version);
|
|
|
|
RARCH_LOG("%s: %u\n",
|
|
|
|
msg_hash_to_str(MSG_COMPILED_AGAINST_API),
|
|
|
|
RETRO_API_VERSION);
|
2016-05-08 01:33:57 +02:00
|
|
|
|
|
|
|
if (api_version != RETRO_API_VERSION)
|
|
|
|
{
|
|
|
|
RARCH_WARN("%s\n", msg_hash_to_str(MSG_LIBRETRO_ABI_BREAK));
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2016-09-03 17:50:04 +01:00
|
|
|
bool core_get_region(retro_ctx_region_info_t *info)
|
|
|
|
{
|
|
|
|
if (!info)
|
|
|
|
return false;
|
|
|
|
info->region = core.retro_get_region();
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2016-05-08 01:33:57 +02:00
|
|
|
bool core_has_set_input_descriptor(void)
|
|
|
|
{
|
|
|
|
return core_has_set_input_descriptors;
|
|
|
|
}
|
|
|
|
|
|
|
|
void core_set_input_descriptors(void)
|
|
|
|
{
|
|
|
|
core_has_set_input_descriptors = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
void core_unset_input_descriptors(void)
|
|
|
|
{
|
|
|
|
core_has_set_input_descriptors = false;
|
|
|
|
}
|
2016-09-03 17:50:04 +01:00
|
|
|
|
2016-09-06 03:52:29 +02:00
|
|
|
bool core_is_inited(void)
|
|
|
|
{
|
|
|
|
return core_inited;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool core_is_symbols_inited(void)
|
|
|
|
{
|
|
|
|
return core_symbols_inited;
|
|
|
|
}
|
|
|
|
|
2016-09-03 17:50:04 +01:00
|
|
|
bool core_is_game_loaded(void)
|
|
|
|
{
|
|
|
|
return core_game_loaded;
|
|
|
|
}
|