mirror of
https://github.com/libretro/RetroArch
synced 2025-04-04 22:20:25 +00:00
Real, actual (almost) >2-player support.
This commit is contained in:
parent
2cc8c5c467
commit
b51cf8be2b
@ -400,18 +400,17 @@ static void send_input(netplay_t *netplay, struct netplay_connection *connection
|
|||||||
}
|
}
|
||||||
|
|
||||||
/* Send a specified input frame */
|
/* Send a specified input frame */
|
||||||
static void send_input_frame(netplay_t *netplay, uint32_t frame, uint32_t *state)
|
static void send_input_frame(netplay_t *netplay,
|
||||||
{
|
struct netplay_connection *except, uint32_t frame, uint32_t player,
|
||||||
if (netplay->self_mode == NETPLAY_CONNECTION_PLAYING)
|
uint32_t *state)
|
||||||
{
|
{
|
||||||
uint32_t buffer[2 + WORDS_PER_FRAME];
|
uint32_t buffer[2 + WORDS_PER_FRAME];
|
||||||
size_t i;
|
size_t i;
|
||||||
|
|
||||||
buffer[0] = htonl(NETPLAY_CMD_INPUT);
|
buffer[0] = htonl(NETPLAY_CMD_INPUT);
|
||||||
buffer[1] = htonl(WORDS_PER_FRAME * sizeof(uint32_t));
|
buffer[1] = htonl(WORDS_PER_FRAME * sizeof(uint32_t));
|
||||||
buffer[2] = htonl(netplay->self_frame_count);
|
buffer[2] = htonl(frame);
|
||||||
buffer[3] = htonl(netplay->self_player |
|
buffer[3] = htonl(player);
|
||||||
(netplay->is_server ? NETPLAY_CMD_INPUT_BIT_SERVER : 0));
|
|
||||||
buffer[4] = htonl(state[0]);
|
buffer[4] = htonl(state[0]);
|
||||||
buffer[5] = htonl(state[1]);
|
buffer[5] = htonl(state[1]);
|
||||||
buffer[6] = htonl(state[2]);
|
buffer[6] = htonl(state[2]);
|
||||||
@ -419,6 +418,7 @@ static void send_input_frame(netplay_t *netplay, uint32_t frame, uint32_t *state
|
|||||||
for (i = 0; i < netplay->connections_size; i++)
|
for (i = 0; i < netplay->connections_size; i++)
|
||||||
{
|
{
|
||||||
struct netplay_connection *connection = &netplay->connections[i];
|
struct netplay_connection *connection = &netplay->connections[i];
|
||||||
|
if (connection == except) continue;
|
||||||
if (connection->active && connection->mode >= NETPLAY_CONNECTION_CONNECTED)
|
if (connection->active && connection->mode >= NETPLAY_CONNECTION_CONNECTED)
|
||||||
{
|
{
|
||||||
if (!netplay_send(&connection->send_packet_buffer, connection->fd,
|
if (!netplay_send(&connection->send_packet_buffer, connection->fd,
|
||||||
@ -427,7 +427,6 @@ static void send_input_frame(netplay_t *netplay, uint32_t frame, uint32_t *state
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* get_self_input_state:
|
* get_self_input_state:
|
||||||
@ -439,7 +438,7 @@ static void send_input_frame(netplay_t *netplay, uint32_t frame, uint32_t *state
|
|||||||
**/
|
**/
|
||||||
static bool get_self_input_state(netplay_t *netplay)
|
static bool get_self_input_state(netplay_t *netplay)
|
||||||
{
|
{
|
||||||
uint32_t state[WORDS_PER_FRAME - 1] = {0, 0, 0};
|
uint32_t state[WORDS_PER_INPUT] = {0, 0, 0};
|
||||||
struct delta_frame *ptr = &netplay->buffer[netplay->self_ptr];
|
struct delta_frame *ptr = &netplay->buffer[netplay->self_ptr];
|
||||||
size_t i;
|
size_t i;
|
||||||
|
|
||||||
@ -733,12 +732,16 @@ static bool netplay_get_cmd(netplay_t *netplay,
|
|||||||
/* FIXME: Catastrophe! */
|
/* FIXME: Catastrophe! */
|
||||||
return netplay_cmd_nak(netplay, connection);
|
return netplay_cmd_nak(netplay, connection);
|
||||||
}
|
}
|
||||||
dframe->have_remote[player] = true;
|
|
||||||
memcpy(dframe->remote_input_state[player], buffer + 2,
|
memcpy(dframe->remote_input_state[player], buffer + 2,
|
||||||
WORDS_PER_INPUT*sizeof(uint32_t));
|
WORDS_PER_INPUT*sizeof(uint32_t));
|
||||||
|
dframe->have_remote[player] = true;
|
||||||
netplay->read_ptr[player] = NEXT_PTR(netplay->read_ptr[player]);
|
netplay->read_ptr[player] = NEXT_PTR(netplay->read_ptr[player]);
|
||||||
netplay->read_frame_count[player]++;
|
netplay->read_frame_count[player]++;
|
||||||
|
|
||||||
|
if (netplay->is_server)
|
||||||
|
/* Forward it on */
|
||||||
|
send_input_frame(netplay, connection, buffer[0], player, buffer + 2);
|
||||||
|
|
||||||
/* If this was server data, advance our server pointer too */
|
/* If this was server data, advance our server pointer too */
|
||||||
if (!netplay->is_server && (buffer[1] & NETPLAY_CMD_INPUT_BIT_SERVER))
|
if (!netplay->is_server && (buffer[1] & NETPLAY_CMD_INPUT_BIT_SERVER))
|
||||||
{
|
{
|
||||||
@ -820,12 +823,26 @@ static bool netplay_get_cmd(netplay_t *netplay,
|
|||||||
case NETPLAY_CMD_PLAY:
|
case NETPLAY_CMD_PLAY:
|
||||||
{
|
{
|
||||||
uint32_t payload[2];
|
uint32_t payload[2];
|
||||||
uint32_t player = 1; /* FIXME */
|
uint32_t player = 0;
|
||||||
payload[0] = htonl(netplay->self_frame_count + 1);
|
payload[0] = htonl(netplay->self_frame_count + 1);
|
||||||
|
|
||||||
if (!netplay->is_server)
|
if (!netplay->is_server)
|
||||||
return netplay_cmd_nak(netplay, connection);
|
return netplay_cmd_nak(netplay, connection);
|
||||||
|
|
||||||
|
/* Find an available player slot */
|
||||||
|
for (player = 0; player < MAX_USERS; player++)
|
||||||
|
{
|
||||||
|
if (!(netplay->self_mode == NETPLAY_CONNECTION_PLAYING &&
|
||||||
|
netplay->self_player == player) &&
|
||||||
|
!(netplay->connected_players & player))
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
if (player == MAX_USERS)
|
||||||
|
{
|
||||||
|
/* FIXME */
|
||||||
|
return netplay_cmd_nak(netplay, connection);
|
||||||
|
}
|
||||||
|
|
||||||
if (connection->mode != NETPLAY_CONNECTION_PLAYING)
|
if (connection->mode != NETPLAY_CONNECTION_PLAYING)
|
||||||
{
|
{
|
||||||
/* Mark them as playing */
|
/* Mark them as playing */
|
||||||
@ -913,7 +930,7 @@ static bool netplay_get_cmd(netplay_t *netplay,
|
|||||||
{
|
{
|
||||||
memcpy(dframe->remote_input_state[player], dframe->self_state, sizeof(dframe->self_state));
|
memcpy(dframe->remote_input_state[player], dframe->self_state, sizeof(dframe->self_state));
|
||||||
dframe->have_remote[player] = true;
|
dframe->have_remote[player] = true;
|
||||||
send_input_frame(netplay, frame, dframe->self_state);
|
send_input_frame(netplay, NULL, dframe->frame, player, dframe->self_state);
|
||||||
if (dframe->frame == netplay->self_frame_count) break;
|
if (dframe->frame == netplay->self_frame_count) break;
|
||||||
NEXT();
|
NEXT();
|
||||||
}
|
}
|
||||||
@ -2163,6 +2180,8 @@ void netplay_load_savestate(netplay_t *netplay,
|
|||||||
return;
|
return;
|
||||||
|
|
||||||
/* Compress it */
|
/* Compress it */
|
||||||
|
if (!netplay->compression_backend)
|
||||||
|
return;
|
||||||
netplay->compression_backend->set_in(netplay->compression_stream,
|
netplay->compression_backend->set_in(netplay->compression_stream,
|
||||||
(const uint8_t*)serial_info->data_const, serial_info->size);
|
(const uint8_t*)serial_info->data_const, serial_info->size);
|
||||||
netplay->compression_backend->set_out(netplay->compression_stream,
|
netplay->compression_backend->set_out(netplay->compression_stream,
|
||||||
|
@ -386,6 +386,7 @@ bool netplay_handshake_pre_sync(netplay_t *netplay, struct netplay_connection *c
|
|||||||
netplay->self_frame_count = netplay->other_frame_count =
|
netplay->self_frame_count = netplay->other_frame_count =
|
||||||
netplay->unread_frame_count = netplay->server_frame_count =
|
netplay->unread_frame_count = netplay->server_frame_count =
|
||||||
netplay->read_frame_count[0] = new_frame_count;
|
netplay->read_frame_count[0] = new_frame_count;
|
||||||
|
netplay->connected_players = 1;
|
||||||
for (i = 0; i < netplay->buffer_size; i++)
|
for (i = 0; i < netplay->buffer_size; i++)
|
||||||
{
|
{
|
||||||
struct delta_frame *ptr = &netplay->buffer[i];
|
struct delta_frame *ptr = &netplay->buffer[i];
|
||||||
@ -443,7 +444,6 @@ bool netplay_handshake_pre_sync(netplay_t *netplay, struct netplay_connection *c
|
|||||||
/* We're ready! */
|
/* We're ready! */
|
||||||
netplay->self_mode = NETPLAY_CONNECTION_SPECTATING;
|
netplay->self_mode = NETPLAY_CONNECTION_SPECTATING;
|
||||||
connection->mode = NETPLAY_CONNECTION_PLAYING;
|
connection->mode = NETPLAY_CONNECTION_PLAYING;
|
||||||
netplay->connected_players = 1;
|
|
||||||
netplay_handshake_ready(netplay, connection);
|
netplay_handshake_ready(netplay, connection);
|
||||||
*had_input = true;
|
*had_input = true;
|
||||||
netplay_recv_flush(&connection->recv_packet_buffer);
|
netplay_recv_flush(&connection->recv_packet_buffer);
|
||||||
|
Loading…
x
Reference in New Issue
Block a user