| 
									
										
										
										
											2015-02-02 01:56:53 -08:00
										 |  |  | // This file is public domain, in case it's useful to anyone. -comex
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-02-02 02:08:58 -08:00
										 |  |  | #include "Common/TraversalClient.h"
 | 
					
						
							| 
									
										
										
										
											2015-09-26 17:13:07 -04:00
										 |  |  | #include "Common/Logging/Log.h"
 | 
					
						
							| 
									
										
										
										
											2016-06-24 10:43:46 +02:00
										 |  |  | #include "Common/MsgHandler.h"
 | 
					
						
							|  |  |  | #include "Common/Timer.h"
 | 
					
						
							| 
									
										
										
										
											2015-02-02 01:56:53 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  | static void GetRandomishBytes(u8* buf, size_t size) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2016-06-24 10:43:46 +02:00
										 |  |  |   // We don't need high quality random numbers (which might not be available),
 | 
					
						
							|  |  |  |   // just non-repeating numbers!
 | 
					
						
							|  |  |  |   static std::mt19937 prng(enet_time_get()); | 
					
						
							|  |  |  |   static std::uniform_int_distribution<unsigned int> u8_distribution(0, 255); | 
					
						
							|  |  |  |   for (size_t i = 0; i < size; i++) | 
					
						
							|  |  |  |     buf[i] = u8_distribution(prng); | 
					
						
							| 
									
										
										
										
											2015-02-02 01:56:53 -08:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-02-14 19:51:08 -08:00
										 |  |  | TraversalClient::TraversalClient(ENetHost* netHost, const std::string& server, const u16 port) | 
					
						
							| 
									
										
										
										
											2016-06-24 10:43:46 +02:00
										 |  |  |     : m_NetHost(netHost), m_Client(nullptr), m_FailureReason(0), m_ConnectRequestId(0), | 
					
						
							|  |  |  |       m_PendingConnect(false), m_Server(server), m_port(port), m_PingTime(0) | 
					
						
							| 
									
										
										
										
											2015-02-02 01:56:53 -08:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2016-06-24 10:43:46 +02:00
										 |  |  |   netHost->intercept = TraversalClient::InterceptCallback; | 
					
						
							| 
									
										
										
										
											2015-02-02 01:56:53 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-06-24 10:43:46 +02:00
										 |  |  |   Reset(); | 
					
						
							| 
									
										
										
										
											2015-02-02 01:56:53 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-06-24 10:43:46 +02:00
										 |  |  |   ReconnectToServer(); | 
					
						
							| 
									
										
										
										
											2015-02-02 01:56:53 -08:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | TraversalClient::~TraversalClient() | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void TraversalClient::ReconnectToServer() | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2016-06-24 10:43:46 +02:00
										 |  |  |   if (enet_address_set_host(&m_ServerAddress, m_Server.c_str())) | 
					
						
							|  |  |  |   { | 
					
						
							|  |  |  |     OnFailure(BadHost); | 
					
						
							|  |  |  |     return; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   m_ServerAddress.port = m_port; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   m_State = Connecting; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   TraversalPacket hello = {}; | 
					
						
							|  |  |  |   hello.type = TraversalPacketHelloFromClient; | 
					
						
							|  |  |  |   hello.helloFromClient.protoVersion = TraversalProtoVersion; | 
					
						
							|  |  |  |   SendTraversalPacket(hello); | 
					
						
							|  |  |  |   if (m_Client) | 
					
						
							|  |  |  |     m_Client->OnTraversalStateChanged(); | 
					
						
							| 
									
										
										
										
											2015-02-02 01:56:53 -08:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static ENetAddress MakeENetAddress(TraversalInetAddress* address) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2016-06-24 10:43:46 +02:00
										 |  |  |   ENetAddress eaddr; | 
					
						
							|  |  |  |   if (address->isIPV6) | 
					
						
							|  |  |  |   { | 
					
						
							|  |  |  |     eaddr.port = 0;  // no support yet :(
 | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   else | 
					
						
							|  |  |  |   { | 
					
						
							|  |  |  |     eaddr.host = address->address[0]; | 
					
						
							|  |  |  |     eaddr.port = ntohs(address->port); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   return eaddr; | 
					
						
							| 
									
										
										
										
											2015-02-02 01:56:53 -08:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void TraversalClient::ConnectToClient(const std::string& host) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2016-06-24 10:43:46 +02:00
										 |  |  |   if (host.size() > sizeof(TraversalHostId)) | 
					
						
							|  |  |  |   { | 
					
						
							|  |  |  |     PanicAlert("host too long"); | 
					
						
							|  |  |  |     return; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   TraversalPacket packet = {}; | 
					
						
							|  |  |  |   packet.type = TraversalPacketConnectPlease; | 
					
						
							|  |  |  |   memcpy(packet.connectPlease.hostId.data(), host.c_str(), host.size()); | 
					
						
							|  |  |  |   m_ConnectRequestId = SendTraversalPacket(packet); | 
					
						
							|  |  |  |   m_PendingConnect = true; | 
					
						
							| 
									
										
										
										
											2015-02-02 01:56:53 -08:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool TraversalClient::TestPacket(u8* data, size_t size, ENetAddress* from) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2016-06-24 10:43:46 +02:00
										 |  |  |   if (from->host == m_ServerAddress.host && from->port == m_ServerAddress.port) | 
					
						
							|  |  |  |   { | 
					
						
							|  |  |  |     if (size < sizeof(TraversalPacket)) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       ERROR_LOG(NETPLAY, "Received too-short traversal packet."); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     else | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       HandleServerPacket((TraversalPacket*)data); | 
					
						
							|  |  |  |       return true; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   return false; | 
					
						
							| 
									
										
										
										
											2015-02-02 01:56:53 -08:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | //--Temporary until more of the old netplay branch is moved over
 | 
					
						
							|  |  |  | void TraversalClient::Update() | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2016-06-24 10:43:46 +02:00
										 |  |  |   ENetEvent netEvent; | 
					
						
							|  |  |  |   if (enet_host_service(m_NetHost, &netEvent, 4) > 0) | 
					
						
							|  |  |  |   { | 
					
						
							|  |  |  |     switch (netEvent.type) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |     case ENET_EVENT_TYPE_RECEIVE: | 
					
						
							|  |  |  |       TestPacket(netEvent.packet->data, netEvent.packet->dataLength, &netEvent.peer->address); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       enet_packet_destroy(netEvent.packet); | 
					
						
							|  |  |  |       break; | 
					
						
							|  |  |  |     default: | 
					
						
							|  |  |  |       break; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   HandleResends(); | 
					
						
							| 
									
										
										
										
											2015-02-02 01:56:53 -08:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void TraversalClient::HandleServerPacket(TraversalPacket* packet) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2016-06-24 10:43:46 +02:00
										 |  |  |   u8 ok = 1; | 
					
						
							|  |  |  |   switch (packet->type) | 
					
						
							|  |  |  |   { | 
					
						
							|  |  |  |   case TraversalPacketAck: | 
					
						
							|  |  |  |     if (!packet->ack.ok) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       OnFailure(ServerForgotAboutUs); | 
					
						
							|  |  |  |       break; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     for (auto it = m_OutgoingTraversalPackets.begin(); it != m_OutgoingTraversalPackets.end(); ++it) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       if (it->packet.requestId == packet->requestId) | 
					
						
							|  |  |  |       { | 
					
						
							|  |  |  |         m_OutgoingTraversalPackets.erase(it); | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     break; | 
					
						
							|  |  |  |   case TraversalPacketHelloFromServer: | 
					
						
							|  |  |  |     if (m_State != Connecting) | 
					
						
							|  |  |  |       break; | 
					
						
							|  |  |  |     if (!packet->helloFromServer.ok) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       OnFailure(VersionTooOld); | 
					
						
							|  |  |  |       break; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     m_HostId = packet->helloFromServer.yourHostId; | 
					
						
							|  |  |  |     m_State = Connected; | 
					
						
							|  |  |  |     if (m_Client) | 
					
						
							|  |  |  |       m_Client->OnTraversalStateChanged(); | 
					
						
							|  |  |  |     break; | 
					
						
							|  |  |  |   case TraversalPacketPleaseSendPacket: | 
					
						
							|  |  |  |   { | 
					
						
							|  |  |  |     // security is overrated.
 | 
					
						
							|  |  |  |     ENetAddress addr = MakeENetAddress(&packet->pleaseSendPacket.address); | 
					
						
							|  |  |  |     if (addr.port != 0) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       char message[] = "Hello from Dolphin Netplay..."; | 
					
						
							|  |  |  |       ENetBuffer buf; | 
					
						
							|  |  |  |       buf.data = message; | 
					
						
							|  |  |  |       buf.dataLength = sizeof(message) - 1; | 
					
						
							|  |  |  |       enet_socket_send(m_NetHost->socket, &addr, &buf, 1); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     else | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       // invalid IPV6
 | 
					
						
							|  |  |  |       ok = 0; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     break; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   case TraversalPacketConnectReady: | 
					
						
							|  |  |  |   case TraversalPacketConnectFailed: | 
					
						
							|  |  |  |   { | 
					
						
							|  |  |  |     if (!m_PendingConnect || packet->connectReady.requestId != m_ConnectRequestId) | 
					
						
							|  |  |  |       break; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     m_PendingConnect = false; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (!m_Client) | 
					
						
							|  |  |  |       break; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (packet->type == TraversalPacketConnectReady) | 
					
						
							|  |  |  |       m_Client->OnConnectReady(MakeENetAddress(&packet->connectReady.address)); | 
					
						
							|  |  |  |     else | 
					
						
							|  |  |  |       m_Client->OnConnectFailed(packet->connectFailed.reason); | 
					
						
							|  |  |  |     break; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   default: | 
					
						
							|  |  |  |     WARN_LOG(NETPLAY, "Received unknown packet with type %d", packet->type); | 
					
						
							|  |  |  |     break; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   if (packet->type != TraversalPacketAck) | 
					
						
							|  |  |  |   { | 
					
						
							|  |  |  |     TraversalPacket ack = {}; | 
					
						
							|  |  |  |     ack.type = TraversalPacketAck; | 
					
						
							|  |  |  |     ack.requestId = packet->requestId; | 
					
						
							|  |  |  |     ack.ack.ok = ok; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     ENetBuffer buf; | 
					
						
							|  |  |  |     buf.data = &ack; | 
					
						
							|  |  |  |     buf.dataLength = sizeof(ack); | 
					
						
							|  |  |  |     if (enet_socket_send(m_NetHost->socket, &m_ServerAddress, &buf, 1) == -1) | 
					
						
							|  |  |  |       OnFailure(SocketSendError); | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2015-02-02 01:56:53 -08:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-02-16 22:53:50 -08:00
										 |  |  | void TraversalClient::OnFailure(FailureReason reason) | 
					
						
							| 
									
										
										
										
											2015-02-02 01:56:53 -08:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2016-06-24 10:43:46 +02:00
										 |  |  |   m_State = Failure; | 
					
						
							|  |  |  |   m_FailureReason = reason; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (m_Client) | 
					
						
							|  |  |  |     m_Client->OnTraversalStateChanged(); | 
					
						
							| 
									
										
										
										
											2015-02-02 01:56:53 -08:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void TraversalClient::ResendPacket(OutgoingTraversalPacketInfo* info) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2016-06-24 10:43:46 +02:00
										 |  |  |   info->sendTime = enet_time_get(); | 
					
						
							|  |  |  |   info->tries++; | 
					
						
							|  |  |  |   ENetBuffer buf; | 
					
						
							|  |  |  |   buf.data = &info->packet; | 
					
						
							|  |  |  |   buf.dataLength = sizeof(info->packet); | 
					
						
							|  |  |  |   if (enet_socket_send(m_NetHost->socket, &m_ServerAddress, &buf, 1) == -1) | 
					
						
							|  |  |  |     OnFailure(SocketSendError); | 
					
						
							| 
									
										
										
										
											2015-02-02 01:56:53 -08:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void TraversalClient::HandleResends() | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2016-06-24 10:43:46 +02:00
										 |  |  |   enet_uint32 now = enet_time_get(); | 
					
						
							|  |  |  |   for (auto& tpi : m_OutgoingTraversalPackets) | 
					
						
							|  |  |  |   { | 
					
						
							|  |  |  |     if (now - tpi.sendTime >= (u32)(300 * tpi.tries)) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       if (tpi.tries >= 5) | 
					
						
							|  |  |  |       { | 
					
						
							|  |  |  |         OnFailure(ResendTimeout); | 
					
						
							|  |  |  |         m_OutgoingTraversalPackets.clear(); | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       else | 
					
						
							|  |  |  |       { | 
					
						
							|  |  |  |         ResendPacket(&tpi); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   HandlePing(); | 
					
						
							| 
									
										
										
										
											2015-02-02 01:56:53 -08:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void TraversalClient::HandlePing() | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2016-06-24 10:43:46 +02:00
										 |  |  |   enet_uint32 now = enet_time_get(); | 
					
						
							|  |  |  |   if (m_State == Connected && now - m_PingTime >= 500) | 
					
						
							|  |  |  |   { | 
					
						
							|  |  |  |     TraversalPacket ping = {}; | 
					
						
							|  |  |  |     ping.type = TraversalPacketPing; | 
					
						
							|  |  |  |     ping.ping.hostId = m_HostId; | 
					
						
							|  |  |  |     SendTraversalPacket(ping); | 
					
						
							|  |  |  |     m_PingTime = now; | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2015-02-02 01:56:53 -08:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | TraversalRequestId TraversalClient::SendTraversalPacket(const TraversalPacket& packet) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2016-06-24 10:43:46 +02:00
										 |  |  |   OutgoingTraversalPacketInfo info; | 
					
						
							|  |  |  |   info.packet = packet; | 
					
						
							|  |  |  |   GetRandomishBytes((u8*)&info.packet.requestId, sizeof(info.packet.requestId)); | 
					
						
							|  |  |  |   info.tries = 0; | 
					
						
							|  |  |  |   m_OutgoingTraversalPackets.push_back(info); | 
					
						
							|  |  |  |   ResendPacket(&m_OutgoingTraversalPackets.back()); | 
					
						
							|  |  |  |   return info.packet.requestId; | 
					
						
							| 
									
										
										
										
											2015-02-02 01:56:53 -08:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void TraversalClient::Reset() | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2016-06-24 10:43:46 +02:00
										 |  |  |   m_PendingConnect = false; | 
					
						
							|  |  |  |   m_Client = nullptr; | 
					
						
							| 
									
										
										
										
											2015-02-02 01:56:53 -08:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | int ENET_CALLBACK TraversalClient::InterceptCallback(ENetHost* host, ENetEvent* event) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2016-06-24 10:43:46 +02:00
										 |  |  |   auto traversalClient = g_TraversalClient.get(); | 
					
						
							|  |  |  |   if (traversalClient->TestPacket(host->receivedData, host->receivedDataLength, | 
					
						
							|  |  |  |                                   &host->receivedAddress) || | 
					
						
							|  |  |  |       (host->receivedDataLength == 1 && host->receivedData[0] == 0)) | 
					
						
							|  |  |  |   { | 
					
						
							|  |  |  |     event->type = (ENetEventType)42; | 
					
						
							|  |  |  |     return 1; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   return 0; | 
					
						
							| 
									
										
										
										
											2015-02-02 01:56:53 -08:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | std::unique_ptr<TraversalClient> g_TraversalClient; | 
					
						
							|  |  |  | std::unique_ptr<ENetHost> g_MainNetHost; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // The settings at the previous TraversalClient reset - notably, we
 | 
					
						
							|  |  |  | // need to know not just what port it's on, but whether it was
 | 
					
						
							|  |  |  | // explicitly requested.
 | 
					
						
							|  |  |  | static std::string g_OldServer; | 
					
						
							| 
									
										
										
										
											2015-09-20 20:08:37 +09:00
										 |  |  | static u16 g_OldServerPort; | 
					
						
							|  |  |  | static u16 g_OldListenPort; | 
					
						
							| 
									
										
										
										
											2015-02-02 01:56:53 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-09-20 20:08:37 +09:00
										 |  |  | bool EnsureTraversalClient(const std::string& server, u16 server_port, u16 listen_port) | 
					
						
							| 
									
										
										
										
											2015-02-02 01:56:53 -08:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2016-06-24 10:43:46 +02:00
										 |  |  |   if (!g_MainNetHost || !g_TraversalClient || server != g_OldServer || | 
					
						
							|  |  |  |       server_port != g_OldServerPort || listen_port != g_OldListenPort) | 
					
						
							|  |  |  |   { | 
					
						
							|  |  |  |     g_OldServer = server; | 
					
						
							|  |  |  |     g_OldServerPort = server_port; | 
					
						
							|  |  |  |     g_OldListenPort = listen_port; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     ENetAddress addr = {ENET_HOST_ANY, listen_port}; | 
					
						
							|  |  |  |     ENetHost* host = enet_host_create(&addr,  // address
 | 
					
						
							|  |  |  |                                       50,     // peerCount
 | 
					
						
							|  |  |  |                                       1,      // channelLimit
 | 
					
						
							|  |  |  |                                       0,      // incomingBandwidth
 | 
					
						
							|  |  |  |                                       0);     // outgoingBandwidth
 | 
					
						
							|  |  |  |     if (!host) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       g_MainNetHost.reset(); | 
					
						
							|  |  |  |       return false; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     g_MainNetHost.reset(host); | 
					
						
							|  |  |  |     g_TraversalClient.reset(new TraversalClient(g_MainNetHost.get(), server, server_port)); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   return true; | 
					
						
							| 
									
										
										
										
											2015-02-02 01:56:53 -08:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void ReleaseTraversalClient() | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2016-06-24 10:43:46 +02:00
										 |  |  |   if (!g_TraversalClient) | 
					
						
							|  |  |  |     return; | 
					
						
							| 
									
										
										
										
											2015-02-02 01:56:53 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-06-24 10:43:46 +02:00
										 |  |  |   g_TraversalClient.release(); | 
					
						
							|  |  |  |   g_MainNetHost.release(); | 
					
						
							| 
									
										
										
										
											2015-03-01 16:17:32 +01:00
										 |  |  | } |