RetroArch/core_impl.c

533 lines
12 KiB
C
Raw Normal View History

/* RetroArch - A frontend for libretro.
* Copyright (C) 2010-2014 - Hans-Kristian Arntzen
2017-03-22 02:09:18 +00:00
* Copyright (C) 2011-2017 - Daniel De Matteis
2015-01-07 16:46:50 +00:00
* Copyright (C) 2012-2015 - Michael Lelli
2017-03-22 02:09:18 +00:00
* Copyright (C) 2016-2017 - Brad Parker
*
* 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>
#include <boolean.h>
#include <lists/string_list.h>
2017-01-24 15:08:25 +00:00
#include <string/stdstring.h>
#include <libretro.h>
2016-09-06 21:53:44 +00:00
#ifdef HAVE_CONFIG_H
#include "config.h"
#endif
2016-09-29 19:07:10 +00:00
#ifdef HAVE_NETWORKING
2016-09-08 04:07:43 +00:00
#include "network/netplay/netplay.h"
#endif
2017-01-23 15:17:57 +00:00
#include "core.h"
2017-01-24 23:34:58 +00:00
#include "content.h"
2016-03-22 01:56:06 +00:00
#include "dynamic.h"
2016-01-27 07:13:54 +00:00
#include "msg_hash.h"
#include "managers/state_manager.h"
2016-09-01 03:36:52 +00:00
#include "verbosity.h"
2015-11-24 00:26:59 +00:00
#include "gfx/video_driver.h"
2015-11-23 18:40:09 +00:00
#include "audio/audio_driver.h"
#include "tasks/tasks_internal.h"
2018-03-29 13:38:22 +00:00
#ifdef HAVE_RUNAHEAD
2018-03-28 19:22:07 +00:00
#include "runahead/copy_load_info.h"
#include "runahead/secondary_core.h"
2018-03-29 13:38:22 +00:00
#endif
2018-03-28 19:22:07 +00:00
struct retro_callbacks retro_ctx;
struct retro_core_t current_core;
static void retro_run_null(void)
{
}
static void retro_frame_null(const void *data, unsigned width,
unsigned height, size_t pitch)
{
}
static void retro_input_poll_null(void)
{
}
static void core_input_state_poll_maybe(void)
{
if (current_core.poll_type == POLL_TYPE_NORMAL)
input_poll();
}
static int16_t core_input_state_poll(unsigned port,
unsigned device, unsigned idx, unsigned id)
{
if (current_core.poll_type == POLL_TYPE_LATE)
{
if (!current_core.input_polled)
input_poll();
current_core.input_polled = true;
}
return input_state(port, device, idx, id);
}
2016-05-07 23:33:57 +00:00
void core_set_input_state(retro_ctx_input_state_info_t *info)
2015-12-05 15:54:03 +00:00
{
current_core.retro_set_input_state(info->cb);
2015-12-05 15:54:03 +00:00
}
2015-01-08 17:39:17 +00:00
/**
* core_init_libretro_cbs:
2015-01-08 17:39:17 +00:00
* @data : pointer to retro_callbacks object
*
* Initializes libretro callbacks, and binds the libretro callbacks
2015-01-08 17:39:17 +00:00
* to default callback functions.
**/
2017-05-21 08:31:25 +00:00
static bool core_init_libretro_cbs(struct retro_callbacks *cbs)
{
current_core.retro_set_video_refresh(video_driver_frame);
current_core.retro_set_audio_sample(audio_driver_sample);
current_core.retro_set_audio_sample_batch(audio_driver_sample_batch);
current_core.retro_set_input_state(core_input_state_poll);
current_core.retro_set_input_poll(core_input_state_poll_maybe);
2016-05-07 23:33:57 +00:00
core_set_default_callbacks(cbs);
2016-09-29 19:07:10 +00:00
#ifdef HAVE_NETWORKING
if (!netplay_driver_ctl(RARCH_NETPLAY_CTL_IS_DATA_INITED, NULL))
return true;
core_set_netplay_callbacks();
#endif
return true;
}
2016-05-07 23:33:57 +00:00
/**
* core_set_default_callbacks:
* @data : pointer to retro_callbacks object
*
* Binds the libretro callbacks to default callback functions.
**/
2017-05-21 08:34:50 +00:00
bool core_set_default_callbacks(struct retro_callbacks *cbs)
2016-05-07 23:33:57 +00:00
{
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 = retro_frame_null;
2016-05-07 23:33:57 +00:00
cbs->sample_cb = NULL;
cbs->sample_batch_cb = NULL;
cbs->state_cb = NULL;
cbs->poll_cb = retro_input_poll_null;
2016-05-07 23:33:57 +00:00
current_core.inited = false;
2016-05-07 23:33:57 +00:00
return true;
}
bool core_uninit_libretro_callbacks(void)
{
return core_deinit(&retro_ctx);
}
2015-01-08 17:39:17 +00:00
/**
* core_set_rewind_callbacks:
2015-01-08 17:39:17 +00:00
*
* Sets the audio sampling callbacks based on whether or not
* rewinding is currently activated.
**/
2016-05-07 23:33:57 +00:00
bool core_set_rewind_callbacks(void)
{
if (state_manager_frame_is_reversed())
{
current_core.retro_set_audio_sample(audio_driver_sample_rewind);
current_core.retro_set_audio_sample_batch(audio_driver_sample_batch_rewind);
}
else
{
current_core.retro_set_audio_sample(audio_driver_sample);
current_core.retro_set_audio_sample_batch(audio_driver_sample_batch);
}
2016-05-07 23:33:57 +00:00
return true;
}
#ifdef HAVE_NETWORKING
/**
* core_set_netplay_callbacks:
*
* Set the I/O callbacks to use netplay's interceding callback system. Should
* only be called while initializing netplay.
**/
bool core_set_netplay_callbacks(void)
{
/* Force normal poll type for netplay. */
current_core.poll_type = POLL_TYPE_NORMAL;
/* And use netplay's interceding callbacks */
current_core.retro_set_video_refresh(video_frame_net);
current_core.retro_set_audio_sample(audio_sample_net);
current_core.retro_set_audio_sample_batch(audio_sample_batch_net);
current_core.retro_set_input_state(input_state_net);
return true;
}
/**
* core_unset_netplay_callbacks
*
* Unset the I/O callbacks from having used netplay's interceding callback
* system. Should only be called while uninitializing netplay.
*/
bool core_unset_netplay_callbacks(void)
{
struct retro_callbacks cbs;
if (!core_set_default_callbacks(&cbs))
return false;
current_core.retro_set_video_refresh(cbs.frame_cb);
current_core.retro_set_audio_sample(cbs.sample_cb);
current_core.retro_set_audio_sample_batch(cbs.sample_batch_cb);
current_core.retro_set_input_state(cbs.state_cb);
return true;
}
#endif
2016-05-07 23:33:57 +00:00
bool core_set_cheat(retro_ctx_cheat_info_t *info)
{
current_core.retro_cheat_set(info->index, info->enabled, info->code);
2016-05-07 23:33:57 +00:00
return true;
}
2016-01-27 02:08:49 +00:00
2016-05-07 23:33:57 +00:00
bool core_reset_cheat(void)
2016-01-27 02:08:49 +00:00
{
current_core.retro_cheat_reset();
2016-05-07 23:33:57 +00:00
return true;
}
2016-05-07 23:33:57 +00:00
bool core_api_version(retro_ctx_api_info_t *api)
{
if (!api)
return false;
api->version = current_core.retro_api_version();
2016-05-07 23:33:57 +00:00
return true;
}
bool core_set_poll_type(unsigned *type)
{
current_core.poll_type = *type;
2016-05-07 23:33:57 +00:00
return true;
}
2016-07-23 05:06:32 +00:00
void core_uninit_symbols(void)
{
uninit_libretro_sym(&current_core);
current_core.symbols_inited = false;
}
2016-05-07 23:33:57 +00:00
bool core_init_symbols(enum rarch_core_type *type)
{
2017-05-21 08:34:50 +00:00
if (!type || !init_libretro_sym(*type, &current_core))
return false;
if (!current_core.retro_run)
current_core.retro_run = retro_run_null;
current_core.symbols_inited = true;
2016-05-07 23:33:57 +00:00
return true;
}
bool core_set_controller_port_device(retro_ctx_controller_info_t *pad)
{
if (!pad)
return false;
2018-03-28 19:22:07 +00:00
2018-03-29 13:38:22 +00:00
#ifdef HAVE_RUNAHEAD
2018-03-28 19:22:07 +00:00
remember_controller_port_device(pad->port, pad->device);
2018-03-29 13:38:22 +00:00
#endif
2018-03-28 19:22:07 +00:00
current_core.retro_set_controller_port_device(pad->port, pad->device);
2016-05-07 23:33:57 +00:00
return true;
}
bool core_get_memory(retro_ctx_memory_info_t *info)
{
if (!info)
return false;
info->size = current_core.retro_get_memory_size(info->id);
info->data = current_core.retro_get_memory_data(info->id);
2016-05-07 23:33:57 +00:00
return true;
}
bool core_load_game(retro_ctx_load_content_info_t *load_info)
{
2017-01-25 14:44:21 +00:00
bool contentless = false;
bool is_inited = false;
2017-01-25 14:44:21 +00:00
video_driver_set_cached_frame_ptr(NULL);
2018-03-29 13:38:22 +00:00
#ifdef HAVE_RUNAHEAD
2018-03-29 13:15:47 +00:00
set_load_content_info(load_info);
clear_controller_port_map();
2018-03-29 13:38:22 +00:00
#endif
2018-03-29 13:15:47 +00:00
content_get_status(&contentless, &is_inited);
set_save_state_in_background(false);
2017-01-25 14:44:21 +00:00
2016-09-11 17:46:13 +00:00
if (load_info && load_info->special)
current_core.game_loaded = current_core.retro_load_game_special(
2016-09-29 18:35:26 +00:00
load_info->special->id, load_info->info, load_info->content->size);
2017-01-24 15:08:25 +00:00
else if (load_info && !string_is_empty(load_info->content->elems[0].data))
current_core.game_loaded = current_core.retro_load_game(load_info->info);
2017-01-25 14:44:21 +00:00
else if (contentless)
current_core.game_loaded = current_core.retro_load_game(NULL);
2017-01-24 23:34:58 +00:00
else
current_core.game_loaded = false;
return current_core.game_loaded;
2016-05-07 23:33:57 +00:00
}
bool core_get_system_info(struct retro_system_info *system)
{
if (!system)
return false;
current_core.retro_get_system_info(system);
2016-05-07 23:33:57 +00:00
return true;
}
bool core_unserialize(retro_ctx_serialize_info_t *info)
{
if (!info || !current_core.retro_unserialize(info->data_const, info->size))
2016-05-07 23:33:57 +00:00
return false;
#if HAVE_NETWORKING
netplay_driver_ctl(RARCH_NETPLAY_CTL_LOAD_SAVESTATE, info);
#endif
2016-05-07 23:33:57 +00:00
return true;
}
bool core_serialize(retro_ctx_serialize_info_t *info)
{
if (!info || !current_core.retro_serialize(info->data, info->size))
2016-05-07 23:33:57 +00:00
return false;
return true;
}
bool core_serialize_size(retro_ctx_size_info_t *info)
{
if (!info)
return false;
info->size = current_core.retro_serialize_size();
2016-05-07 23:33:57 +00:00
return true;
}
uint64_t core_serialization_quirks(void)
{
return current_core.serialization_quirks_v;
}
void core_set_serialization_quirks(uint64_t quirks)
{
current_core.serialization_quirks_v = quirks;
}
2016-05-07 23:33:57 +00:00
bool core_set_environment(retro_ctx_environ_info_t *info)
{
if (!info)
return false;
current_core.retro_set_environment(info->env);
2016-05-07 23:33:57 +00:00
return true;
}
bool core_get_system_av_info(struct retro_system_av_info *av_info)
{
if (!av_info)
return false;
current_core.retro_get_system_av_info(av_info);
2016-05-07 23:33:57 +00:00
return true;
}
bool core_reset(void)
{
video_driver_set_cached_frame_ptr(NULL);
current_core.retro_reset();
2016-05-07 23:33:57 +00:00
return true;
}
bool core_init(void)
{
video_driver_set_cached_frame_ptr(NULL);
current_core.retro_init();
current_core.inited = true;
2016-05-07 23:33:57 +00:00
return true;
}
bool core_unload(void)
{
video_driver_set_cached_frame_ptr(NULL);
if (current_core.inited)
current_core.retro_deinit();
2016-05-07 23:33:57 +00:00
return true;
}
bool core_unload_game(void)
{
2017-01-25 14:22:12 +00:00
video_driver_free_hw_context();
video_driver_set_cached_frame_ptr(NULL);
if (current_core.game_loaded)
{
current_core.retro_unload_game();
current_core.game_loaded = false;
}
audio_driver_stop();
2016-05-07 23:33:57 +00:00
return true;
}
bool core_run(void)
{
#ifdef HAVE_NETWORKING
if (!netplay_driver_ctl(RARCH_NETPLAY_CTL_PRE_FRAME, NULL))
{
/* Paused due to netplay. We must poll and display something so that a
* netplay peer pausing doesn't just hang. */
input_poll();
video_driver_cached_frame();
return true;
}
#endif
switch (current_core.poll_type)
2016-01-27 02:08:49 +00:00
{
2016-05-07 23:33:57 +00:00
case POLL_TYPE_EARLY:
input_poll();
2016-04-05 22:20:14 +00:00
break;
2016-05-07 23:33:57 +00:00
case POLL_TYPE_LATE:
current_core.input_polled = false;
2016-01-27 02:08:49 +00:00
break;
2016-09-29 18:35:26 +00:00
default:
break;
2016-01-27 02:08:49 +00:00
}
2016-09-29 18:35:26 +00:00
current_core.retro_run();
if (current_core.poll_type == POLL_TYPE_LATE && !current_core.input_polled)
2016-05-07 23:33:57 +00:00
input_poll();
#ifdef HAVE_NETWORKING
netplay_driver_ctl(RARCH_NETPLAY_CTL_POST_FRAME, NULL);
#endif
2016-05-07 23:33:57 +00:00
return true;
}
2018-03-28 19:22:07 +00:00
bool core_run_no_input_polling(void)
{
current_core.retro_run();
return true;
}
2017-01-23 13:50:00 +00:00
bool core_load(unsigned poll_type_behavior)
2016-05-07 23:33:57 +00:00
{
current_core.poll_type = poll_type_behavior;
2016-05-07 23:33:57 +00:00
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());
2016-01-27 02:08:49 +00:00
return true;
}
2016-05-07 23:33:57 +00:00
bool core_verify_api_version(void)
{
unsigned api_version = current_core.retro_api_version();
RARCH_LOG("%s: %u\n",
2016-06-20 02:23:00 +00:00
msg_hash_to_str(MSG_VERSION_OF_LIBRETRO_API),
api_version);
RARCH_LOG("%s: %u\n",
2016-06-20 02:23:00 +00:00
msg_hash_to_str(MSG_COMPILED_AGAINST_API),
RETRO_API_VERSION);
2016-05-07 23:33:57 +00:00
if (api_version != RETRO_API_VERSION)
{
RARCH_WARN("%s\n", msg_hash_to_str(MSG_LIBRETRO_ABI_BREAK));
return false;
}
return true;
}
bool core_get_region(retro_ctx_region_info_t *info)
{
if (!info)
return false;
info->region = current_core.retro_get_region();
return true;
}
2016-05-07 23:33:57 +00:00
bool core_has_set_input_descriptor(void)
{
return current_core.has_set_input_descriptors;
2016-05-07 23:33:57 +00:00
}
void core_set_input_descriptors(void)
{
current_core.has_set_input_descriptors = true;
2016-05-07 23:33:57 +00:00
}
void core_unset_input_descriptors(void)
{
current_core.has_set_input_descriptors = false;
2016-05-07 23:33:57 +00:00
}
bool core_is_inited(void)
{
return current_core.inited;
}
bool core_is_symbols_inited(void)
{
return current_core.symbols_inited;
}
bool core_is_game_loaded(void)
{
return current_core.game_loaded;
}