RetroArch/network/netplay_net.c

170 lines
4.4 KiB
C
Raw Normal View History

2015-12-23 20:25:28 +00:00
/* RetroArch - A frontend for libretro.
* Copyright (C) 2010-2014 - Hans-Kristian Arntzen
2016-01-10 03:06:50 +00:00
* Copyright (C) 2011-2016 - Daniel De Matteis
2015-12-23 20:25:28 +00: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 "netplay_private.h"
/**
* pre_frame:
* @netplay : pointer to netplay object
*
* Pre-frame for Netplay (normal version).
**/
2015-12-24 18:23:46 +00:00
static void netplay_net_pre_frame(netplay_t *netplay)
2015-12-23 20:25:28 +00:00
{
retro_ctx_serialize_info_t serial_info;
serial_info.data = netplay->buffer[netplay->self_ptr].state;
serial_info.size = netplay->state_size;
2016-05-07 23:33:57 +00:00
core_serialize(&serial_info);
2015-12-23 20:25:28 +00:00
netplay->can_poll = true;
input_poll_net();
}
/**
* post_frame:
* @netplay : pointer to netplay object
*
* Post-frame for Netplay (normal version).
* We check if we have new input and replay from recorded input.
**/
2015-12-24 18:23:46 +00:00
static void netplay_net_post_frame(netplay_t *netplay)
2015-12-23 20:25:28 +00:00
{
netplay->frame_count++;
/* Nothing to do... */
if (netplay->other_frame_count == netplay->read_frame_count)
return;
/* Skip ahead if we predicted correctly.
* Skip until our simulation failed. */
while (netplay->other_frame_count < netplay->read_frame_count)
{
const struct delta_frame *ptr = &netplay->buffer[netplay->other_ptr];
2016-02-03 15:58:30 +00:00
if (memcmp(ptr->simulated_input_state, ptr->real_input_state,
sizeof(ptr->real_input_state)) != 0
2015-12-23 20:25:28 +00:00
&& !ptr->used_real)
break;
netplay->other_ptr = NEXT_PTR(netplay->other_ptr);
netplay->other_frame_count++;
}
if (netplay->other_frame_count < netplay->read_frame_count)
{
retro_ctx_serialize_info_t serial_info;
2015-12-23 20:25:28 +00:00
bool first = true;
/* Replay frames. */
netplay->is_replay = true;
netplay->tmp_ptr = netplay->other_ptr;
netplay->tmp_frame_count = netplay->other_frame_count;
serial_info.data_const = netplay->buffer[netplay->other_ptr].state;
serial_info.size = netplay->state_size;
2016-05-07 23:33:57 +00:00
core_unserialize(&serial_info);
2015-12-23 20:25:28 +00:00
while (first || (netplay->tmp_ptr != netplay->self_ptr))
{
serial_info.data = netplay->buffer[netplay->tmp_ptr].state;
serial_info.size = netplay->state_size;
2016-04-10 14:35:25 +00:00
serial_info.data_const = NULL;
2016-05-07 23:33:57 +00:00
core_serialize(&serial_info);
#if defined(HAVE_THREADS)
2016-05-09 06:17:35 +00:00
autosave_lock();
2015-12-23 20:25:28 +00:00
#endif
2016-05-07 23:33:57 +00:00
core_run();
#if defined(HAVE_THREADS)
2016-05-09 06:17:35 +00:00
autosave_unlock();
2015-12-23 20:25:28 +00:00
#endif
netplay->tmp_ptr = NEXT_PTR(netplay->tmp_ptr);
netplay->tmp_frame_count++;
first = false;
}
netplay->other_ptr = netplay->read_ptr;
netplay->other_frame_count = netplay->read_frame_count;
netplay->is_replay = false;
}
}
2015-12-24 18:23:46 +00:00
static bool netplay_net_init_buffers(netplay_t *netplay)
2015-12-23 20:25:28 +00:00
{
unsigned i;
retro_ctx_size_info_t info;
2015-12-23 20:25:28 +00:00
if (!netplay)
return false;
netplay->buffer = (struct delta_frame*)calloc(netplay->buffer_size,
sizeof(*netplay->buffer));
if (!netplay->buffer)
return false;
2016-05-07 23:33:57 +00:00
core_serialize_size(&info);
netplay->state_size = info.size;
2015-12-23 20:25:28 +00:00
for (i = 0; i < netplay->buffer_size; i++)
{
netplay->buffer[i].state = malloc(netplay->state_size);
if (!netplay->buffer[i].state)
return false;
netplay->buffer[i].is_simulated = true;
}
return true;
}
2015-12-24 18:23:46 +00:00
static bool netplay_net_info_cb(netplay_t* netplay, unsigned frames)
2015-12-23 20:25:28 +00:00
{
if (netplay_is_server(netplay))
2015-12-23 20:25:28 +00:00
{
2016-05-12 10:03:43 +00:00
if (!netplay_send_info(netplay))
2015-12-23 20:25:28 +00:00
return false;
}
else
{
2016-05-12 10:03:43 +00:00
if (!netplay_get_info(netplay))
2015-12-23 20:25:28 +00:00
return false;
}
netplay->buffer_size = frames + 1;
2015-12-24 18:23:46 +00:00
if (!netplay_net_init_buffers(netplay))
2015-12-23 20:25:28 +00:00
return false;
netplay->has_connection = true;
return true;
}
struct netplay_callbacks* netplay_get_cbs_net(void)
{
static struct netplay_callbacks cbs = {
2015-12-24 18:23:46 +00:00
&netplay_net_pre_frame,
&netplay_net_post_frame,
&netplay_net_info_cb
2015-12-23 20:25:28 +00:00
};
return &cbs;
2016-01-10 03:06:50 +00:00
}