2013-06-29 18:57:18 -04:00
|
|
|
//
|
|
|
|
// SuperTuxKart - a fun racing game with go-kart
|
2015-03-29 20:31:42 -04:00
|
|
|
// Copyright (C) 2013-2015 SuperTuxKart-Team
|
2013-06-29 18:57:18 -04:00
|
|
|
//
|
|
|
|
// This program 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 Foundation; either version 3
|
|
|
|
// of the License, or (at your option) any later version.
|
|
|
|
//
|
|
|
|
// This program 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 this program; if not, write to the Free Software
|
|
|
|
// Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
|
|
|
|
|
2013-07-04 09:19:32 -04:00
|
|
|
#include "network/stk_peer.hpp"
|
2018-08-03 07:50:49 -04:00
|
|
|
#include "config/stk_config.hpp"
|
2018-03-12 01:59:38 -04:00
|
|
|
#include "config/user_config.hpp"
|
2018-06-02 00:28:29 -04:00
|
|
|
#include "network/crypto.hpp"
|
|
|
|
#include "network/event.hpp"
|
2018-08-03 07:50:49 -04:00
|
|
|
#include "network/network_config.hpp"
|
2015-10-22 01:03:11 -04:00
|
|
|
#include "network/network_string.hpp"
|
2016-03-11 00:42:35 -05:00
|
|
|
#include "network/stk_host.hpp"
|
2015-10-29 03:39:01 -04:00
|
|
|
#include "network/transport_address.hpp"
|
2013-07-04 12:59:30 -04:00
|
|
|
#include "utils/log.hpp"
|
2018-03-04 12:54:44 -05:00
|
|
|
#include "utils/time.hpp"
|
2013-07-04 12:59:30 -04:00
|
|
|
|
2013-06-29 18:57:18 -04:00
|
|
|
#include <string.h>
|
|
|
|
|
2015-10-22 01:03:11 -04:00
|
|
|
/** Constructor for an empty peer.
|
|
|
|
*/
|
2018-03-11 12:18:53 -04:00
|
|
|
STKPeer::STKPeer(ENetPeer *enet_peer, STKHost* host, uint32_t host_id)
|
2018-03-10 01:56:32 -05:00
|
|
|
: m_peer_address(enet_peer->address), m_host(host)
|
2013-06-29 18:57:18 -04:00
|
|
|
{
|
2015-11-16 16:11:07 -05:00
|
|
|
m_enet_peer = enet_peer;
|
2018-03-11 12:18:53 -04:00
|
|
|
m_host_id = host_id;
|
2018-08-27 01:49:11 -04:00
|
|
|
m_connected_time = StkTime::getRealTimeMs();
|
2018-06-04 01:25:10 -04:00
|
|
|
m_validated.store(false);
|
2018-08-26 21:16:35 -04:00
|
|
|
m_average_ping.store(0);
|
2018-08-31 04:27:32 -04:00
|
|
|
m_waiting_for_game.store(true);
|
2018-09-12 22:46:37 -04:00
|
|
|
m_disconnected.store(false);
|
2018-12-04 19:35:11 -05:00
|
|
|
m_warned_for_high_ping.store(false);
|
2018-12-04 22:12:44 -05:00
|
|
|
m_last_activity.store((int64_t)StkTime::getRealTimeMs());
|
2015-10-22 01:03:11 -04:00
|
|
|
} // STKPeer
|
2013-07-11 08:03:28 -04:00
|
|
|
|
2018-06-02 00:28:29 -04:00
|
|
|
//-----------------------------------------------------------------------------
|
|
|
|
STKPeer::~STKPeer()
|
|
|
|
{
|
|
|
|
} // ~STKPeer
|
|
|
|
|
2013-07-11 08:03:28 -04:00
|
|
|
//-----------------------------------------------------------------------------
|
2018-03-11 12:18:53 -04:00
|
|
|
void STKPeer::disconnect()
|
2013-06-29 18:57:18 -04:00
|
|
|
{
|
2018-03-09 23:34:33 -05:00
|
|
|
TransportAddress a(m_enet_peer->address);
|
|
|
|
if (m_enet_peer->state != ENET_PEER_STATE_CONNECTED ||
|
|
|
|
a != m_peer_address)
|
|
|
|
return;
|
2018-09-12 22:46:37 -04:00
|
|
|
m_disconnected.store(true);
|
2018-03-10 01:56:32 -05:00
|
|
|
m_host->addEnetCommand(m_enet_peer, NULL, PDI_NORMAL, ECT_DISCONNECT);
|
2018-03-11 12:18:53 -04:00
|
|
|
} // disconnect
|
2013-06-29 18:57:18 -04:00
|
|
|
|
2013-07-11 08:03:28 -04:00
|
|
|
//-----------------------------------------------------------------------------
|
2018-03-09 23:34:33 -05:00
|
|
|
/** Kick this peer (used by server).
|
2015-10-22 01:03:11 -04:00
|
|
|
*/
|
2018-03-09 23:34:33 -05:00
|
|
|
void STKPeer::kick()
|
2013-07-09 19:35:53 -04:00
|
|
|
{
|
2018-03-09 23:34:33 -05:00
|
|
|
TransportAddress a(m_enet_peer->address);
|
|
|
|
if (m_enet_peer->state != ENET_PEER_STATE_CONNECTED ||
|
|
|
|
a != m_peer_address)
|
|
|
|
return;
|
2018-09-12 22:46:37 -04:00
|
|
|
m_disconnected.store(true);
|
2018-03-10 01:56:32 -05:00
|
|
|
m_host->addEnetCommand(m_enet_peer, NULL, PDI_KICK, ECT_DISCONNECT);
|
2018-03-09 23:34:33 -05:00
|
|
|
} // kick
|
2013-07-09 19:35:53 -04:00
|
|
|
|
2018-03-11 12:18:53 -04:00
|
|
|
//-----------------------------------------------------------------------------
|
|
|
|
/** Forcefully disconnects a peer (used by server).
|
|
|
|
*/
|
|
|
|
void STKPeer::reset()
|
|
|
|
{
|
|
|
|
TransportAddress a(m_enet_peer->address);
|
|
|
|
if (m_enet_peer->state != ENET_PEER_STATE_CONNECTED ||
|
|
|
|
a != m_peer_address)
|
|
|
|
return;
|
2018-09-12 22:46:37 -04:00
|
|
|
m_disconnected.store(true);
|
2018-03-11 12:18:53 -04:00
|
|
|
m_host->addEnetCommand(m_enet_peer, NULL, 0, ECT_RESET);
|
|
|
|
} // reset
|
|
|
|
|
2013-07-11 08:03:28 -04:00
|
|
|
//-----------------------------------------------------------------------------
|
2015-10-22 01:03:11 -04:00
|
|
|
/** Sends a packet to this host.
|
|
|
|
* \param data The data to send.
|
|
|
|
* \param reliable If the data is sent reliable or not.
|
2018-06-02 00:28:29 -04:00
|
|
|
* \param encrypted If the data is sent encrypted or not.
|
2015-10-22 01:03:11 -04:00
|
|
|
*/
|
2018-06-02 00:28:29 -04:00
|
|
|
void STKPeer::sendPacket(NetworkString *data, bool reliable, bool encrypted)
|
2013-06-29 18:57:18 -04:00
|
|
|
{
|
2015-10-22 01:03:11 -04:00
|
|
|
TransportAddress a(m_enet_peer->address);
|
2018-03-09 12:07:23 -05:00
|
|
|
// Enet will reuse a disconnected peer so we check here to avoid sending
|
|
|
|
// to wrong peer
|
|
|
|
if (m_enet_peer->state != ENET_PEER_STATE_CONNECTED ||
|
|
|
|
a != m_peer_address)
|
|
|
|
return;
|
2018-03-09 05:33:19 -05:00
|
|
|
|
2018-06-02 00:28:29 -04:00
|
|
|
ENetPacket* packet = NULL;
|
|
|
|
if (m_crypto && encrypted)
|
|
|
|
{
|
|
|
|
packet = m_crypto->encryptSend(*data, reliable);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
packet = enet_packet_create(data->getData(),
|
|
|
|
data->getTotalSize(), (reliable ?
|
2018-07-27 09:47:07 -04:00
|
|
|
ENET_PACKET_FLAG_RELIABLE :
|
|
|
|
(ENET_PACKET_FLAG_UNSEQUENCED |
|
|
|
|
ENET_PACKET_FLAG_UNRELIABLE_FRAGMENT)));
|
2018-06-02 00:28:29 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
if (packet)
|
|
|
|
{
|
2018-06-08 00:12:59 -04:00
|
|
|
if (Network::m_connection_debug)
|
|
|
|
{
|
2018-08-27 01:49:11 -04:00
|
|
|
Log::verbose("STKPeer", "sending packet of size %d to %s at %lf",
|
2018-06-08 00:12:59 -04:00
|
|
|
packet->dataLength, a.toString().c_str(),
|
|
|
|
StkTime::getRealTime());
|
|
|
|
}
|
2018-06-02 00:28:29 -04:00
|
|
|
m_host->addEnetCommand(m_enet_peer, packet,
|
|
|
|
encrypted ? EVENT_CHANNEL_NORMAL : EVENT_CHANNEL_UNENCRYPTED,
|
|
|
|
ECT_SEND_PACKET);
|
|
|
|
}
|
2015-10-22 01:03:11 -04:00
|
|
|
} // sendPacket
|
2013-06-29 18:57:18 -04:00
|
|
|
|
2013-07-11 08:03:28 -04:00
|
|
|
//-----------------------------------------------------------------------------
|
2015-10-22 01:03:11 -04:00
|
|
|
/** Returns if the peer is connected or not.
|
|
|
|
*/
|
2013-07-07 15:49:51 -04:00
|
|
|
bool STKPeer::isConnected() const
|
2013-06-29 18:57:18 -04:00
|
|
|
{
|
2018-03-09 23:34:33 -05:00
|
|
|
Log::debug("STKPeer", "The peer state is %i", m_enet_peer->state);
|
2015-10-22 01:03:11 -04:00
|
|
|
return (m_enet_peer->state == ENET_PEER_STATE_CONNECTED);
|
|
|
|
} // isConnected
|
2013-07-11 08:03:28 -04:00
|
|
|
|
2013-07-29 19:43:28 -04:00
|
|
|
//-----------------------------------------------------------------------------
|
2015-11-16 16:11:07 -05:00
|
|
|
/** Returns if this STKPeer is the same as the given peer.
|
|
|
|
*/
|
2013-07-10 20:03:47 -04:00
|
|
|
bool STKPeer::isSamePeer(const STKPeer* peer) const
|
2013-07-10 19:03:45 -04:00
|
|
|
{
|
2015-10-22 01:03:11 -04:00
|
|
|
return peer->m_enet_peer==m_enet_peer;
|
2015-11-16 16:11:07 -05:00
|
|
|
} // isSamePeer
|
|
|
|
|
|
|
|
//-----------------------------------------------------------------------------
|
|
|
|
/** Returns if this STKPeer is the same as the given peer.
|
|
|
|
*/
|
|
|
|
bool STKPeer::isSamePeer(const ENetPeer* peer) const
|
|
|
|
{
|
|
|
|
return peer==m_enet_peer;
|
|
|
|
} // isSamePeer
|
2018-03-23 20:20:27 -04:00
|
|
|
|
|
|
|
//-----------------------------------------------------------------------------
|
2018-05-21 13:31:08 -04:00
|
|
|
/** Returns the ping to this peer from host, it waits for 3 seconds for a
|
2018-03-23 20:20:27 -04:00
|
|
|
* stable ping returned by enet measured in ms.
|
|
|
|
*/
|
2018-08-03 07:50:49 -04:00
|
|
|
uint32_t STKPeer::getPing()
|
2018-03-23 20:20:27 -04:00
|
|
|
{
|
2018-08-27 01:49:11 -04:00
|
|
|
if (getConnectedTime() < 3.0f)
|
2018-08-27 01:49:52 -04:00
|
|
|
{
|
|
|
|
m_average_ping.store(m_enet_peer->roundTripTime);
|
2018-03-23 20:20:27 -04:00
|
|
|
return 0;
|
2018-08-27 01:49:52 -04:00
|
|
|
}
|
2018-08-03 07:50:49 -04:00
|
|
|
if (NetworkConfig::get()->isServer())
|
|
|
|
{
|
|
|
|
// Average ping in 5 seconds
|
|
|
|
const unsigned ap = stk_config->m_network_state_frequeny * 5;
|
|
|
|
m_previous_pings.push_back(m_enet_peer->roundTripTime);
|
|
|
|
while (m_previous_pings.size() > ap)
|
|
|
|
{
|
|
|
|
m_previous_pings.pop_front();
|
2018-08-26 21:16:35 -04:00
|
|
|
m_average_ping.store(
|
2018-08-03 07:50:49 -04:00
|
|
|
(uint32_t)(std::accumulate(m_previous_pings.begin(),
|
2018-08-26 21:16:35 -04:00
|
|
|
m_previous_pings.end(), 0) / m_previous_pings.size()));
|
2018-08-03 07:50:49 -04:00
|
|
|
}
|
|
|
|
}
|
2018-05-21 13:31:08 -04:00
|
|
|
return m_enet_peer->roundTripTime;
|
2018-03-23 20:20:27 -04:00
|
|
|
} // getPing
|
2018-06-02 00:28:29 -04:00
|
|
|
|
|
|
|
//-----------------------------------------------------------------------------
|
|
|
|
void STKPeer::setCrypto(std::unique_ptr<Crypto>&& c)
|
|
|
|
{
|
|
|
|
m_crypto = std::move(c);
|
|
|
|
} // setCrypto
|