|
|
|
/*==LICENSE==*
|
|
|
|
|
|
|
|
CyanWorlds.com Engine - MMOG client, server and tools
|
|
|
|
Copyright (C) 2011 Cyan Worlds, Inc.
|
|
|
|
|
|
|
|
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, see <http://www.gnu.org/licenses/>.
|
|
|
|
|
|
|
|
Additional permissions under GNU GPL version 3 section 7
|
|
|
|
|
|
|
|
If you modify this Program, or any covered work, by linking or
|
|
|
|
combining it with any of RAD Game Tools Bink SDK, Autodesk 3ds Max SDK,
|
|
|
|
NVIDIA PhysX SDK, Microsoft DirectX SDK, OpenSSL library, Independent
|
|
|
|
JPEG Group JPEG library, Microsoft Windows Media SDK, or Apple QuickTime SDK
|
|
|
|
(or a modified version of those libraries),
|
|
|
|
containing parts covered by the terms of the Bink SDK EULA, 3ds Max EULA,
|
|
|
|
PhysX SDK EULA, DirectX SDK EULA, OpenSSL and SSLeay licenses, IJG
|
|
|
|
JPEG Library README, Windows Media SDK EULA, or QuickTime SDK EULA, the
|
|
|
|
licensors of this Program grant you additional
|
|
|
|
permission to convey the resulting work. Corresponding Source for a
|
|
|
|
non-source form of such a combination shall include the source code for
|
|
|
|
the parts of OpenSSL and IJG JPEG Library used as well as that of the covered
|
|
|
|
work.
|
|
|
|
|
|
|
|
You can contact Cyan Worlds, Inc. by email legal@cyan.com
|
|
|
|
or by snail mail at:
|
|
|
|
Cyan Worlds, Inc.
|
|
|
|
14617 N Newport Hwy
|
|
|
|
Mead, WA 99021
|
|
|
|
|
|
|
|
*==LICENSE==*/
|
|
|
|
/*****************************************************************************
|
|
|
|
*
|
|
|
|
* $/Plasma20/Sources/Plasma/PubUtilLib/plNetGateKeeperLib/Private/plNglGateKeeper.cpp
|
|
|
|
*
|
|
|
|
***/
|
|
|
|
|
|
|
|
#include "../Pch.h"
|
|
|
|
#pragma hdrstop
|
|
|
|
|
|
|
|
namespace Ngl { namespace GateKeeper {
|
|
|
|
|
|
|
|
/*****************************************************************************
|
|
|
|
*
|
|
|
|
* Private
|
|
|
|
*
|
|
|
|
***/
|
|
|
|
|
|
|
|
struct CliGkConn : hsAtomicRefCnt {
|
|
|
|
CliGkConn ();
|
|
|
|
~CliGkConn ();
|
|
|
|
|
|
|
|
// Reconnection
|
|
|
|
AsyncTimer * reconnectTimer;
|
|
|
|
unsigned reconnectStartMs;
|
|
|
|
|
|
|
|
// Ping
|
|
|
|
AsyncTimer * pingTimer;
|
|
|
|
unsigned pingSendTimeMs;
|
|
|
|
unsigned lastHeardTimeMs;
|
|
|
|
|
|
|
|
// This function should be called during object construction
|
|
|
|
// to initiate connection attempts to the remote host whenever
|
|
|
|
// the socket is disconnected.
|
|
|
|
void AutoReconnect ();
|
|
|
|
bool AutoReconnectEnabled ();
|
|
|
|
void StopAutoReconnect (); // call before destruction
|
|
|
|
void StartAutoReconnect ();
|
|
|
|
void TimerReconnect ();
|
|
|
|
|
|
|
|
// ping
|
|
|
|
void AutoPing ();
|
|
|
|
void StopAutoPing ();
|
|
|
|
void TimerPing ();
|
|
|
|
|
|
|
|
void Send (const uintptr_t fields[], unsigned count);
|
|
|
|
|
|
|
|
CCritSect critsect;
|
|
|
|
LINK(CliGkConn) link;
|
|
|
|
AsyncSocket sock;
|
|
|
|
NetCli * cli;
|
|
|
|
char name[MAX_PATH];
|
|
|
|
plNetAddress addr;
|
|
|
|
plUUID token;
|
|
|
|
unsigned seq;
|
|
|
|
unsigned serverChallenge;
|
|
|
|
AsyncCancelId cancelId;
|
|
|
|
bool abandoned;
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
//============================================================================
|
|
|
|
// PingRequestTrans
|
|
|
|
//============================================================================
|
|
|
|
struct PingRequestTrans : NetGateKeeperTrans {
|
|
|
|
FNetCliGateKeeperPingRequestCallback m_callback;
|
|
|
|
void * m_param;
|
|
|
|
unsigned m_pingAtMs;
|
|
|
|
unsigned m_replyAtMs;
|
|
|
|
ARRAY(uint8_t) m_payload;
|
|
|
|
|
|
|
|
PingRequestTrans (
|
|
|
|
FNetCliGateKeeperPingRequestCallback callback,
|
|
|
|
void * param,
|
|
|
|
unsigned pingAtMs,
|
|
|
|
unsigned payloadBytes,
|
|
|
|
const void * payload
|
|
|
|
);
|
|
|
|
|
|
|
|
bool Send ();
|
|
|
|
void Post ();
|
|
|
|
bool Recv (
|
|
|
|
const uint8_t msg[],
|
|
|
|
unsigned bytes
|
|
|
|
);
|
|
|
|
};
|
|
|
|
|
|
|
|
//============================================================================
|
|
|
|
// FileSrvIpAddressRequestTrans
|
|
|
|
//============================================================================
|
|
|
|
struct FileSrvIpAddressRequestTrans : NetGateKeeperTrans {
|
|
|
|
FNetCliGateKeeperFileSrvIpAddressRequestCallback m_callback;
|
|
|
|
void * m_param;
|
|
|
|
wchar_t m_addr[64];
|
|
|
|
bool m_isPatcher;
|
|
|
|
|
|
|
|
FileSrvIpAddressRequestTrans (
|
|
|
|
FNetCliGateKeeperFileSrvIpAddressRequestCallback callback,
|
|
|
|
void * param,
|
|
|
|
bool isPatcher
|
|
|
|
);
|
|
|
|
|
|
|
|
bool Send ();
|
|
|
|
void Post ();
|
|
|
|
bool Recv (
|
|
|
|
const uint8_t msg[],
|
|
|
|
unsigned bytes
|
|
|
|
);
|
|
|
|
};
|
|
|
|
|
|
|
|
//============================================================================
|
|
|
|
// AuthSrvIpAddressRequestTrans
|
|
|
|
//============================================================================
|
|
|
|
struct AuthSrvIpAddressRequestTrans : NetGateKeeperTrans {
|
|
|
|
FNetCliGateKeeperAuthSrvIpAddressRequestCallback m_callback;
|
|
|
|
void * m_param;
|
|
|
|
wchar_t m_addr[64];
|
|
|
|
|
|
|
|
AuthSrvIpAddressRequestTrans (
|
|
|
|
FNetCliGateKeeperAuthSrvIpAddressRequestCallback callback,
|
|
|
|
void * param
|
|
|
|
);
|
|
|
|
|
|
|
|
bool Send ();
|
|
|
|
void Post ();
|
|
|
|
bool Recv (
|
|
|
|
const uint8_t msg[],
|
|
|
|
unsigned bytes
|
|
|
|
);
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
/*****************************************************************************
|
|
|
|
*
|
|
|
|
* Private data
|
|
|
|
*
|
|
|
|
***/
|
|
|
|
|
|
|
|
enum {
|
|
|
|
kPerfConnCount,
|
|
|
|
kPingDisabled,
|
|
|
|
kAutoReconnectDisabled,
|
|
|
|
kNumPerf
|
|
|
|
};
|
|
|
|
|
|
|
|
static bool s_running;
|
|
|
|
static CCritSect s_critsect;
|
|
|
|
static LISTDECL(CliGkConn, link) s_conns;
|
|
|
|
static CliGkConn * s_active;
|
|
|
|
|
|
|
|
static std::atomic<long> s_perf[kNumPerf];
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/*****************************************************************************
|
|
|
|
*
|
|
|
|
* Internal functions
|
|
|
|
*
|
|
|
|
***/
|
|
|
|
|
|
|
|
//===========================================================================
|
|
|
|
static unsigned GetNonZeroTimeMs () {
|
|
|
|
if (unsigned ms = hsTimer::GetMilliSeconds<uint32_t>())
|
|
|
|
return ms;
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
//============================================================================
|
|
|
|
static CliGkConn * GetConnIncRef_CS (const char tag[]) {
|
|
|
|
if (CliGkConn * conn = s_active) {
|
|
|
|
conn->Ref(tag);
|
|
|
|
return conn;
|
|
|
|
}
|
|
|
|
return nil;
|
|
|
|
}
|
|
|
|
|
|
|
|
//============================================================================
|
|
|
|
static CliGkConn * GetConnIncRef (const char tag[]) {
|
|
|
|
CliGkConn * conn;
|
|
|
|
s_critsect.Enter();
|
|
|
|
{
|
|
|
|
conn = GetConnIncRef_CS(tag);
|
|
|
|
}
|
|
|
|
s_critsect.Leave();
|
|
|
|
return conn;
|
|
|
|
}
|
|
|
|
|
|
|
|
//============================================================================
|
|
|
|
static void UnlinkAndAbandonConn_CS (CliGkConn * conn) {
|
|
|
|
s_conns.Unlink(conn);
|
|
|
|
conn->abandoned = true;
|
|
|
|
|
|
|
|
conn->StopAutoReconnect();
|
|
|
|
|
|
|
|
if (conn->cancelId) {
|
|
|
|
AsyncSocketConnectCancel(nil, conn->cancelId);
|
|
|
|
conn->cancelId = 0;
|
|
|
|
}
|
|
|
|
else if (conn->sock) {
|
|
|
|
AsyncSocketDisconnect(conn->sock, true);
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
conn->UnRef("Lifetime");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
//============================================================================
|
|
|
|
static void SendClientRegisterRequest (CliGkConn * conn) {
|
|
|
|
/* const uintptr_t msg[] = {
|
|
|
|
kCli2GateKeeper_ClientRegisterRequest,
|
|
|
|
BuildId(),
|
|
|
|
};
|
|
|
|
|
|
|
|
conn->Send(msg, arrsize(msg));*/
|
|
|
|
}
|
|
|
|
|
|
|
|
//============================================================================
|
|
|
|
static bool ConnEncrypt (ENetError error, void * param) {
|
|
|
|
CliGkConn * conn = (CliGkConn *) param;
|
|
|
|
|
|
|
|
if (IS_NET_SUCCESS(error)) {
|
|
|
|
|
|
|
|
//SendClientRegisterRequest(conn);
|
|
|
|
|
|
|
|
if (!s_perf[kPingDisabled])
|
|
|
|
conn->AutoPing();
|
|
|
|
|
|
|
|
s_critsect.Enter();
|
|
|
|
{
|
|
|
|
SWAP(s_active, conn);
|
|
|
|
}
|
|
|
|
s_critsect.Leave();
|
|
|
|
|
|
|
|
// AuthConnectedNotifyTrans * trans = new AuthConnectedNotifyTrans;
|
|
|
|
// NetTransSend(trans);
|
|
|
|
}
|
|
|
|
|
|
|
|
return IS_NET_SUCCESS(error);
|
|
|
|
}
|
|
|
|
|
|
|
|
//============================================================================
|
|
|
|
static void NotifyConnSocketConnect (CliGkConn * conn) {
|
|
|
|
|
|
|
|
conn->TransferRef("Connecting", "Connected");
|
|
|
|
conn->cli = NetCliConnectAccept(
|
|
|
|
conn->sock,
|
|
|
|
kNetProtocolCli2GateKeeper,
|
|
|
|
false,
|
|
|
|
ConnEncrypt,
|
|
|
|
0,
|
|
|
|
nil,
|
|
|
|
conn
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
//============================================================================
|
|
|
|
static void CheckedReconnect (CliGkConn * conn, ENetError error) {
|
|
|
|
|
|
|
|
unsigned disconnectedMs = GetNonZeroTimeMs() - conn->lastHeardTimeMs;
|
|
|
|
|
|
|
|
// no auto-reconnect or haven't heard from the server in a while?
|
|
|
|
if (!conn->AutoReconnectEnabled() || (int)disconnectedMs >= (int)kDisconnectedTimeoutMs) {
|
|
|
|
// Cancel all transactions in progress on this connection.
|
|
|
|
NetTransCancelByConnId(conn->seq, kNetErrTimeout);
|
|
|
|
// conn is dead.
|
|
|
|
conn->UnRef("Lifetime");
|
|
|
|
ReportNetError(kNetProtocolCli2GateKeeper, error);
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
if (conn->token != kNilUuid)
|
|
|
|
// previously encrypted; reconnect quickly
|
|
|
|
conn->reconnectStartMs = GetNonZeroTimeMs() + 500;
|
|
|
|
else
|
|
|
|
// never encrypted; reconnect slowly
|
|
|
|
conn->reconnectStartMs = GetNonZeroTimeMs() + kMaxReconnectIntervalMs;
|
|
|
|
|
|
|
|
// clean up the socket and start reconnect
|
|
|
|
if (conn->cli)
|
|
|
|
NetCliDelete(conn->cli, true);
|
|
|
|
conn->cli = nil;
|
|
|
|
conn->sock = nil;
|
|
|
|
|
|
|
|
conn->StartAutoReconnect();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
//============================================================================
|
|
|
|
static void NotifyConnSocketConnectFailed (CliGkConn * conn) {
|
|
|
|
|
|
|
|
s_critsect.Enter();
|
|
|
|
{
|
|
|
|
conn->cancelId = 0;
|
|
|
|
s_conns.Unlink(conn);
|
|
|
|
|
|
|
|
if (conn == s_active)
|
|
|
|
s_active = nil;
|
|
|
|
}
|
|
|
|
s_critsect.Leave();
|
|
|
|
|
|
|
|
CheckedReconnect(conn, kNetErrConnectFailed);
|
|
|
|
|
|
|
|
conn->UnRef("Connecting");
|
|
|
|
}
|
|
|
|
|
|
|
|
//============================================================================
|
|
|
|
static void NotifyConnSocketDisconnect (CliGkConn * conn) {
|
|
|
|
|
|
|
|
conn->StopAutoPing();
|
|
|
|
|
|
|
|
s_critsect.Enter();
|
|
|
|
{
|
|
|
|
conn->cancelId = 0;
|
|
|
|
s_conns.Unlink(conn);
|
|
|
|
|
|
|
|
if (conn == s_active)
|
|
|
|
s_active = nil;
|
|
|
|
}
|
|
|
|
s_critsect.Leave();
|
|
|
|
|
|
|
|
// Cancel all transactions in process on this connection.
|
|
|
|
NetTransCancelByConnId(conn->seq, kNetErrTimeout);
|
|
|
|
conn->UnRef("Connected");
|
|
|
|
conn->UnRef("Lifetime");
|
|
|
|
}
|
|
|
|
|
|
|
|
//============================================================================
|
|
|
|
static bool NotifyConnSocketRead (CliGkConn * conn, AsyncNotifySocketRead * read) {
|
|
|
|
// TODO: Only dispatch messages from the active auth server
|
|
|
|
conn->lastHeardTimeMs = GetNonZeroTimeMs();
|
|
|
|
bool result = NetCliDispatch(conn->cli, read->buffer, read->bytes, conn);
|
|
|
|
read->bytesProcessed += read->bytes;
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
//============================================================================
|
|
|
|
static bool SocketNotifyCallback (
|
|
|
|
AsyncSocket sock,
|
|
|
|
EAsyncNotifySocket code,
|
|
|
|
AsyncNotifySocket * notify,
|
|
|
|
void ** userState
|
|
|
|
) {
|
|
|
|
bool result = true;
|
|
|
|
CliGkConn * conn;
|
|
|
|
|
|
|
|
switch (code) {
|
|
|
|
case kNotifySocketConnectSuccess:
|
|
|
|
conn = (CliGkConn *) notify->param;
|
|
|
|
*userState = conn;
|
|
|
|
bool abandoned;
|
|
|
|
s_critsect.Enter();
|
|
|
|
{
|
|
|
|
conn->sock = sock;
|
|
|
|
conn->cancelId = 0;
|
|
|
|
abandoned = conn->abandoned;
|
|
|
|
}
|
|
|
|
s_critsect.Leave();
|
|
|
|
if (abandoned)
|
|
|
|
AsyncSocketDisconnect(sock, true);
|
|
|
|
else
|
|
|
|
NotifyConnSocketConnect(conn);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case kNotifySocketConnectFailed:
|
|
|
|
conn = (CliGkConn *) notify->param;
|
|
|
|
NotifyConnSocketConnectFailed(conn);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case kNotifySocketDisconnect:
|
|
|
|
conn = (CliGkConn *) *userState;
|
|
|
|
NotifyConnSocketDisconnect(conn);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case kNotifySocketRead:
|
|
|
|
conn = (CliGkConn *) *userState;
|
|
|
|
result = NotifyConnSocketRead(conn, (AsyncNotifySocketRead *) notify);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
//============================================================================
|
|
|
|
static void Connect (
|
|
|
|
CliGkConn * conn
|
|
|
|
) {
|
|
|
|
ASSERT(s_running);
|
|
|
|
|
|
|
|
conn->pingSendTimeMs = 0;
|
|
|
|
|
|
|
|
s_critsect.Enter();
|
|
|
|
{
|
|
|
|
while (CliGkConn * oldConn = s_conns.Head()) {
|
|
|
|
if (oldConn != conn)
|
|
|
|
UnlinkAndAbandonConn_CS(oldConn);
|
|
|
|
else
|
|
|
|
s_conns.Unlink(oldConn);
|
|
|
|
}
|
|
|
|
s_conns.Link(conn);
|
|
|
|
}
|
|
|
|
s_critsect.Leave();
|
|
|
|
|
|
|
|
Cli2GateKeeper_Connect connect;
|
|
|
|
connect.hdr.connType = kConnTypeCliToGateKeeper;
|
|
|
|
connect.hdr.hdrBytes = sizeof(connect.hdr);
|
|
|
|
connect.hdr.buildId = plProduct::BuildId();
|
|
|
|
connect.hdr.buildType = plProduct::BuildType();
|
|
|
|
connect.hdr.branchId = plProduct::BranchId();
|
|
|
|
connect.hdr.productId = plProduct::UUID();
|
|
|
|
connect.data.token = conn->token;
|
|
|
|
connect.data.dataBytes = sizeof(connect.data);
|
|
|
|
|
|
|
|
AsyncSocketConnect(
|
|
|
|
&conn->cancelId,
|
|
|
|
conn->addr,
|
|
|
|
SocketNotifyCallback,
|
|
|
|
conn,
|
|
|
|
&connect,
|
|
|
|
sizeof(connect),
|
|
|
|
0,
|
|
|
|
0
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
//============================================================================
|
|
|
|
static void Connect (
|
|
|
|
const char name[],
|
|
|
|
const plNetAddress& addr
|
|
|
|
) {
|
|
|
|
ASSERT(s_running);
|
|
|
|
|
|
|
|
CliGkConn * conn = new CliGkConn;
|
|
|
|
conn->addr = addr;
|
|
|
|
conn->seq = ConnNextSequence();
|
|
|
|
conn->lastHeardTimeMs = GetNonZeroTimeMs(); // used in connect timeout, and ping timeout
|
|
|
|
strncpy(conn->name, name, arrsize(conn->name));
|
|
|
|
|
|
|
|
conn->Ref("Lifetime");
|
|
|
|
conn->AutoReconnect();
|
|
|
|
}
|
|
|
|
|
|
|
|
//============================================================================
|
|
|
|
static void AsyncLookupCallback (
|
|
|
|
void * param,
|
|
|
|
const char name[],
|
|
|
|
unsigned addrCount,
|
|
|
|
const plNetAddress addrs[]
|
|
|
|
) {
|
|
|
|
if (!addrCount) {
|
|
|
|
ReportNetError(kNetProtocolCli2GateKeeper, kNetErrNameLookupFailed);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
for (unsigned i = 0; i < addrCount; ++i) {
|
|
|
|
Connect(name, addrs[i]);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/*****************************************************************************
|
|
|
|
*
|
|
|
|
* CliGkConn
|
|
|
|
*
|
|
|
|
***/
|
|
|
|
|
|
|
|
//===========================================================================
|
|
|
|
static unsigned CliGkConnTimerDestroyed (void * param) {
|
|
|
|
CliGkConn * conn = (CliGkConn *) param;
|
|
|
|
conn->UnRef("TimerDestroyed");
|
|
|
|
return kAsyncTimeInfinite;
|
|
|
|
}
|
|
|
|
|
|
|
|
//===========================================================================
|
|
|
|
static unsigned CliGkConnReconnectTimerProc (void * param) {
|
|
|
|
((CliGkConn *) param)->TimerReconnect();
|
|
|
|
return kAsyncTimeInfinite;
|
|
|
|
}
|
|
|
|
|
|
|
|
//===========================================================================
|
|
|
|
static unsigned CliGkConnPingTimerProc (void * param) {
|
|
|
|
((CliGkConn *) param)->TimerPing();
|
|
|
|
return kPingIntervalMs;
|
|
|
|
}
|
|
|
|
|
|
|
|
//============================================================================
|
|
|
|
CliGkConn::CliGkConn ()
|
|
|
|
: hsAtomicRefCnt(0), reconnectTimer(nil), reconnectStartMs(0)
|
|
|
|
, pingTimer(nil), pingSendTimeMs(0), lastHeardTimeMs(0)
|
|
|
|
, sock(nil), cli(nil), seq(0), serverChallenge(0)
|
|
|
|
, cancelId(nil), abandoned(false)
|
|
|
|
{
|
|
|
|
memset(name, 0, sizeof(name));
|
|
|
|
|
|
|
|
++s_perf[kPerfConnCount];
|
|
|
|
}
|
|
|
|
|
|
|
|
//============================================================================
|
|
|
|
CliGkConn::~CliGkConn () {
|
|
|
|
if (cli)
|
|
|
|
NetCliDelete(cli, true);
|
|
|
|
--s_perf[kPerfConnCount];
|
|
|
|
}
|
|
|
|
|
|
|
|
//===========================================================================
|
|
|
|
void CliGkConn::TimerReconnect () {
|
|
|
|
ASSERT(!sock);
|
|
|
|
ASSERT(!cancelId);
|
|
|
|
|
|
|
|
if (!s_running) {
|
|
|
|
s_critsect.Enter();
|
|
|
|
UnlinkAndAbandonConn_CS(this);
|
|
|
|
s_critsect.Leave();
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
Ref("Connecting");
|
|
|
|
|
|
|
|
// Remember the time we started the reconnect attempt, guarding against
|
|
|
|
// TimeGetMs() returning zero (unlikely), as a value of zero indicates
|
|
|
|
// a first-time connect condition to StartAutoReconnect()
|
|
|
|
reconnectStartMs = GetNonZeroTimeMs();
|
|
|
|
|
|
|
|
Connect(this);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
//===========================================================================
|
|
|
|
// This function is called when after a disconnect to start a new connection
|
|
|
|
void CliGkConn::StartAutoReconnect () {
|
|
|
|
critsect.Enter();
|
|
|
|
if (reconnectTimer && !s_perf[kAutoReconnectDisabled]) {
|
|
|
|
// Make reconnect attempts at regular intervals. If the last attempt
|
|
|
|
// took more than the specified max interval time then reconnect
|
|
|
|
// immediately; otherwise wait until the time interval is up again
|
|
|
|
// then reconnect.
|
|
|
|
unsigned remainingMs = 0;
|
|
|
|
if (reconnectStartMs) {
|
|
|
|
remainingMs = reconnectStartMs - GetNonZeroTimeMs();
|
|
|
|
if ((signed)remainingMs < 0)
|
|
|
|
remainingMs = 0;
|
|
|
|
LogMsg(kLogPerf, L"GateKeeper auto-reconnecting in %u ms", remainingMs);
|
|
|
|
}
|
|
|
|
AsyncTimerUpdate(reconnectTimer, remainingMs);
|
|
|
|
}
|
|
|
|
critsect.Leave();
|
|
|
|
}
|
|
|
|
|
|
|
|
//===========================================================================
|
|
|
|
// This function should be called during object construction
|
|
|
|
// to initiate connection attempts to the remote host whenever
|
|
|
|
// the socket is disconnected.
|
|
|
|
void CliGkConn::AutoReconnect () {
|
|
|
|
|
|
|
|
ASSERT(!reconnectTimer);
|
|
|
|
Ref("ReconnectTimer");
|
|
|
|
critsect.Enter();
|
|
|
|
{
|
|
|
|
AsyncTimerCreate(
|
|
|
|
&reconnectTimer,
|
|
|
|
CliGkConnReconnectTimerProc,
|
|
|
|
0, // immediate callback
|
|
|
|
this
|
|
|
|
);
|
|
|
|
}
|
|
|
|
critsect.Leave();
|
|
|
|
}
|
|
|
|
|
|
|
|
//============================================================================
|
|
|
|
void CliGkConn::StopAutoReconnect () {
|
|
|
|
critsect.Enter();
|
|
|
|
{
|
|
|
|
if (AsyncTimer * timer = reconnectTimer) {
|
|
|
|
reconnectTimer = nil;
|
|
|
|
AsyncTimerDeleteCallback(timer, CliGkConnTimerDestroyed);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
critsect.Leave();
|
|
|
|
}
|
|
|
|
|
|
|
|
//============================================================================
|
|
|
|
bool CliGkConn::AutoReconnectEnabled () {
|
|
|
|
|
|
|
|
return (reconnectTimer != nil) && !s_perf[kAutoReconnectDisabled];
|
|
|
|
}
|
|
|
|
|
|
|
|
//============================================================================
|
|
|
|
void CliGkConn::AutoPing () {
|
|
|
|
ASSERT(!pingTimer);
|
|
|
|
Ref("PingTimer");
|
|
|
|
critsect.Enter();
|
|
|
|
{
|
|
|
|
AsyncTimerCreate(
|
|
|
|
&pingTimer,
|
|
|
|
CliGkConnPingTimerProc,
|
|
|
|
sock ? 0 : kAsyncTimeInfinite,
|
|
|
|
this
|
|
|
|
);
|
|
|
|
}
|
|
|
|
critsect.Leave();
|
|
|
|
}
|
|
|
|
|
|
|
|
//============================================================================
|
|
|
|
void CliGkConn::StopAutoPing () {
|
|
|
|
critsect.Enter();
|
|
|
|
{
|
|
|
|
if (pingTimer) {
|
|
|
|
AsyncTimerDeleteCallback(pingTimer, CliGkConnTimerDestroyed);
|
|
|
|
pingTimer = nil;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
critsect.Leave();
|
|
|
|
}
|
|
|
|
|
|
|
|
//============================================================================
|
|
|
|
void CliGkConn::TimerPing () {
|
|
|
|
// Send a ping request
|
|
|
|
pingSendTimeMs = GetNonZeroTimeMs();
|
|
|
|
|
|
|
|
const uintptr_t msg[] = {
|
|
|
|
kCli2GateKeeper_PingRequest,
|
|
|
|
pingSendTimeMs,
|
|
|
|
0, // not a transaction
|
|
|
|
0, // no payload
|
|
|
|
reinterpret_cast<uintptr_t>(nullptr)
|
|
|
|
};
|
|
|
|
|
|
|
|
Send(msg, arrsize(msg));
|
|
|
|
}
|
|
|
|
|
|
|
|
//============================================================================
|
|
|
|
void CliGkConn::Send (const uintptr_t fields[], unsigned count) {
|
|
|
|
critsect.Enter();
|
|
|
|
{
|
|
|
|
NetCliSend(cli, fields, count);
|
|
|
|
NetCliFlush(cli);
|
|
|
|
}
|
|
|
|
critsect.Leave();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/*****************************************************************************
|
|
|
|
*
|
|
|
|
* Cli2GateKeeper message handlers
|
|
|
|
*
|
|
|
|
***/
|
|
|
|
|
|
|
|
//============================================================================
|
|
|
|
static bool Recv_PingReply (
|
|
|
|
const uint8_t msg[],
|
|
|
|
unsigned bytes,
|
|
|
|
void *
|
|
|
|
) {
|
|
|
|
const GateKeeper2Cli_PingReply & reply = *(const GateKeeper2Cli_PingReply *)msg;
|
|
|
|
|
|
|
|
if (reply.transId)
|
|
|
|
NetTransRecv(reply.transId, msg, bytes);
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
//============================================================================
|
|
|
|
static bool Recv_FileSrvIpAddressReply (
|
|
|
|
const uint8_t msg[],
|
|
|
|
unsigned bytes,
|
|
|
|
void *
|
|
|
|
) {
|
|
|
|
const GateKeeper2Cli_FileSrvIpAddressReply & reply = *(const GateKeeper2Cli_FileSrvIpAddressReply *)msg;
|
|
|
|
|
|
|
|
if (reply.transId)
|
|
|
|
NetTransRecv(reply.transId, msg, bytes);
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
//============================================================================
|
|
|
|
static bool Recv_AuthSrvIpAddressReply (
|
|
|
|
const uint8_t msg[],
|
|
|
|
unsigned bytes,
|
|
|
|
void *
|
|
|
|
) {
|
|
|
|
const GateKeeper2Cli_AuthSrvIpAddressReply & reply = *(const GateKeeper2Cli_AuthSrvIpAddressReply *)msg;
|
|
|
|
|
|
|
|
if (reply.transId)
|
|
|
|
NetTransRecv(reply.transId, msg, bytes);
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/*****************************************************************************
|
|
|
|
*
|
|
|
|
* Cli2Auth protocol
|
|
|
|
*
|
|
|
|
***/
|
|
|
|
|
|
|
|
#define MSG(s) kNetMsg_Cli2GateKeeper_##s
|
|
|
|
static NetMsgInitSend s_send[] = {
|
|
|
|
{ MSG(PingRequest) },
|
|
|
|
{ MSG(FileSrvIpAddressRequest) },
|
|
|
|
{ MSG(AuthSrvIpAddressRequest) },
|
|
|
|
};
|
|
|
|
#undef MSG
|
|
|
|
|
|
|
|
#define MSG(s) kNetMsg_GateKeeper2Cli_##s, Recv_##s
|
|
|
|
static NetMsgInitRecv s_recv[] = {
|
|
|
|
{ MSG(PingReply) },
|
|
|
|
{ MSG(FileSrvIpAddressReply) },
|
|
|
|
{ MSG(AuthSrvIpAddressReply) },
|
|
|
|
};
|
|
|
|
#undef MSG
|
|
|
|
|
|
|
|
|
|
|
|
/*****************************************************************************
|
|
|
|
*
|
|
|
|
* PingRequestTrans
|
|
|
|
*
|
|
|
|
***/
|
|
|
|
|
|
|
|
//============================================================================
|
|
|
|
PingRequestTrans::PingRequestTrans (
|
|
|
|
FNetCliGateKeeperPingRequestCallback callback,
|
|
|
|
void * param,
|
|
|
|
unsigned pingAtMs,
|
|
|
|
unsigned payloadBytes,
|
|
|
|
const void * payload
|
|
|
|
) : NetGateKeeperTrans(kPingRequestTrans)
|
|
|
|
, m_callback(callback)
|
|
|
|
, m_param(param)
|
|
|
|
, m_pingAtMs(pingAtMs)
|
|
|
|
{
|
|
|
|
m_payload.Set((const uint8_t *)payload, payloadBytes);
|
|
|
|
}
|
|
|
|
|
|
|
|
//============================================================================
|
|
|
|
bool PingRequestTrans::Send () {
|
|
|
|
|
|
|
|
if (!AcquireConn())
|
|
|
|
return false;
|
|
|
|
|
|
|
|
const uintptr_t msg[] = {
|
|
|
|
kCli2GateKeeper_PingRequest,
|
|
|
|
m_pingAtMs,
|
|
|
|
m_transId,
|
|
|
|
m_payload.Count(),
|
|
|
|
(uintptr_t) m_payload.Ptr(),
|
|
|
|
};
|
|
|
|
|
|
|
|
m_conn->Send(msg, arrsize(msg));
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
//============================================================================
|
|
|
|
void PingRequestTrans::Post () {
|
|
|
|
|
|
|
|
m_callback(
|
|
|
|
m_result,
|
|
|
|
m_param,
|
|
|
|
m_pingAtMs,
|
|
|
|
m_replyAtMs,
|
|
|
|
m_payload.Count(),
|
|
|
|
m_payload.Ptr()
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
//============================================================================
|
|
|
|
bool PingRequestTrans::Recv (
|
|
|
|
const uint8_t msg[],
|
|
|
|
unsigned bytes
|
|
|
|
) {
|
|
|
|
const GateKeeper2Cli_PingReply & reply = *(const GateKeeper2Cli_PingReply *)msg;
|
|
|
|
|
|
|
|
m_payload.Set(reply.payload, reply.payloadBytes);
|
|
|
|
m_replyAtMs = hsTimer::GetMilliSeconds<uint32_t>();
|
|
|
|
m_result = kNetSuccess;
|
|
|
|
m_state = kTransStateComplete;
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/*****************************************************************************
|
|
|
|
*
|
|
|
|
* FileSrvIpAddressRequestTrans
|
|
|
|
*
|
|
|
|
***/
|
|
|
|
|
|
|
|
//============================================================================
|
|
|
|
FileSrvIpAddressRequestTrans::FileSrvIpAddressRequestTrans (
|
|
|
|
FNetCliGateKeeperFileSrvIpAddressRequestCallback callback,
|
|
|
|
void * param,
|
|
|
|
bool isPatcher
|
|
|
|
) : NetGateKeeperTrans(kGkFileSrvIpAddressRequestTrans)
|
|
|
|
, m_callback(callback)
|
|
|
|
, m_param(param)
|
|
|
|
, m_isPatcher(isPatcher)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
//============================================================================
|
|
|
|
bool FileSrvIpAddressRequestTrans::Send () {
|
|
|
|
|
|
|
|
if (!AcquireConn())
|
|
|
|
return false;
|
|
|
|
|
|
|
|
|
|
|
|
const uintptr_t msg[] = {
|
|
|
|
kCli2GateKeeper_FileSrvIpAddressRequest,
|
|
|
|
m_transId,
|
|
|
|
(uintptr_t)(m_isPatcher == true ? 1 : 0)
|
|
|
|
};
|
|
|
|
|
|
|
|
m_conn->Send(msg, arrsize(msg));
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
//============================================================================
|
|
|
|
void FileSrvIpAddressRequestTrans::Post () {
|
|
|
|
|
|
|
|
m_callback(
|
|
|
|
m_result,
|
|
|
|
m_param,
|
|
|
|
m_addr
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
//============================================================================
|
|
|
|
bool FileSrvIpAddressRequestTrans::Recv (
|
|
|
|
const uint8_t msg[],
|
|
|
|
unsigned bytes
|
|
|
|
) {
|
|
|
|
const GateKeeper2Cli_FileSrvIpAddressReply & reply = *(const GateKeeper2Cli_FileSrvIpAddressReply *)msg;
|
|
|
|
|
|
|
|
|
|
|
|
m_result = kNetSuccess;
|
|
|
|
m_state = kTransStateComplete;
|
|
|
|
StrCopy(m_addr, reply.address, 64);
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/*****************************************************************************
|
|
|
|
*
|
|
|
|
* AuthSrvIpAddressRequestTrans
|
|
|
|
*
|
|
|
|
***/
|
|
|
|
|
|
|
|
//============================================================================
|
|
|
|
AuthSrvIpAddressRequestTrans::AuthSrvIpAddressRequestTrans (
|
|
|
|
FNetCliGateKeeperFileSrvIpAddressRequestCallback callback,
|
|
|
|
void * param
|
|
|
|
) : NetGateKeeperTrans(kGkAuthSrvIpAddressRequestTrans)
|
|
|
|
, m_callback(callback)
|
|
|
|
, m_param(param)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
//============================================================================
|
|
|
|
bool AuthSrvIpAddressRequestTrans::Send () {
|
|
|
|
|
|
|
|
if (!AcquireConn())
|
|
|
|
return false;
|
|
|
|
|
|
|
|
const uintptr_t msg[] = {
|
|
|
|
kCli2GateKeeper_AuthSrvIpAddressRequest,
|
|
|
|
m_transId,
|
|
|
|
};
|
|
|
|
|
|
|
|
m_conn->Send(msg, arrsize(msg));
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
//============================================================================
|
|
|
|
void AuthSrvIpAddressRequestTrans::Post () {
|
|
|
|
|
|
|
|
m_callback(
|
|
|
|
m_result,
|
|
|
|
m_param,
|
|
|
|
m_addr
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
//============================================================================
|
|
|
|
bool AuthSrvIpAddressRequestTrans::Recv (
|
|
|
|
const uint8_t msg[],
|
|
|
|
unsigned bytes
|
|
|
|
) {
|
|
|
|
const GateKeeper2Cli_AuthSrvIpAddressReply & reply = *(const GateKeeper2Cli_AuthSrvIpAddressReply *)msg;
|
|
|
|
|
|
|
|
m_result = kNetSuccess;
|
|
|
|
m_state = kTransStateComplete;
|
|
|
|
StrCopy(m_addr, reply.address, 64);
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
} using namespace GateKeeper;
|
|
|
|
|
|
|
|
|
|
|
|
/*****************************************************************************
|
|
|
|
*
|
|
|
|
* NetGateKeeperTrans
|
|
|
|
*
|
|
|
|
***/
|
|
|
|
|
|
|
|
//============================================================================
|
|
|
|
NetGateKeeperTrans::NetGateKeeperTrans (ETransType transType)
|
|
|
|
: NetTrans(kNetProtocolCli2GateKeeper, transType)
|
|
|
|
, m_conn(nil)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
//============================================================================
|
|
|
|
NetGateKeeperTrans::~NetGateKeeperTrans () {
|
|
|
|
ReleaseConn();
|
|
|
|
}
|
|
|
|
|
|
|
|
//============================================================================
|
|
|
|
bool NetGateKeeperTrans::AcquireConn () {
|
|
|
|
if (!m_conn)
|
|
|
|
m_conn = GetConnIncRef("AcquireConn");
|
|
|
|
return m_conn != nil;
|
|
|
|
}
|
|
|
|
|
|
|
|
//============================================================================
|
|
|
|
void NetGateKeeperTrans::ReleaseConn () {
|
|
|
|
if (m_conn) {
|
|
|
|
m_conn->UnRef("AcquireConn");
|
|
|
|
m_conn = nil;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/*****************************************************************************
|
|
|
|
*
|
|
|
|
* Protected functions
|
|
|
|
*
|
|
|
|
***/
|
|
|
|
|
|
|
|
//============================================================================
|
|
|
|
void GateKeeperInitialize () {
|
|
|
|
s_running = true;
|
|
|
|
NetMsgProtocolRegister(
|
|
|
|
kNetProtocolCli2GateKeeper,
|
|
|
|
false,
|
|
|
|
s_send, arrsize(s_send),
|
|
|
|
s_recv, arrsize(s_recv),
|
|
|
|
kGateKeeperDhGValue,
|
|
|
|
plBigNum(sizeof(kGateKeeperDhXData), kGateKeeperDhXData),
|
|
|
|
plBigNum(sizeof(kGateKeeperDhNData), kGateKeeperDhNData)
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
//============================================================================
|
|
|
|
void GateKeeperDestroy (bool wait) {
|
|
|
|
s_running = false;
|
|
|
|
|
|
|
|
NetTransCancelByProtocol(
|
|
|
|
kNetProtocolCli2GateKeeper,
|
|
|
|
kNetErrRemoteShutdown
|
|
|
|
);
|
|
|
|
NetMsgProtocolDestroy(
|
|
|
|
kNetProtocolCli2GateKeeper,
|
|
|
|
false
|
|
|
|
);
|
|
|
|
|
|
|
|
s_critsect.Enter();
|
|
|
|
{
|
|
|
|
while (CliGkConn * conn = s_conns.Head())
|
|
|
|
UnlinkAndAbandonConn_CS(conn);
|
|
|
|
s_active = nil;
|
|
|
|
}
|
|
|
|
s_critsect.Leave();
|
|
|
|
|
|
|
|
if (!wait)
|
|
|
|
return;
|
|
|
|
|
|
|
|
while (s_perf[kPerfConnCount]) {
|
|
|
|
NetTransUpdate();
|
|
|
|
AsyncSleep(10);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
//============================================================================
|
|
|
|
bool GateKeeperQueryConnected () {
|
|
|
|
|
|
|
|
bool result;
|
|
|
|
s_critsect.Enter();
|
|
|
|
{
|
|
|
|
result = (s_active && s_active->cli);
|
|
|
|
}
|
|
|
|
s_critsect.Leave();
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
//============================================================================
|
|
|
|
unsigned GateKeeperGetConnId () {
|
|
|
|
unsigned connId;
|
|
|
|
s_critsect.Enter();
|
|
|
|
connId = (s_active) ? s_active->seq : 0;
|
|
|
|
s_critsect.Leave();
|
|
|
|
return connId;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
} using namespace Ngl;
|
|
|
|
|
|
|
|
|
|
|
|
/*****************************************************************************
|
|
|
|
*
|
|
|
|
* Exported functions
|
|
|
|
*
|
|
|
|
***/
|
|
|
|
|
|
|
|
//============================================================================
|
|
|
|
void NetCliGateKeeperStartConnect (
|
|
|
|
const char* gateKeeperAddrList[],
|
|
|
|
uint32_t gateKeeperAddrCount
|
|
|
|
) {
|
|
|
|
gateKeeperAddrCount = min(gateKeeperAddrCount, 1);
|
|
|
|
|
|
|
|
for (unsigned i = 0; i < gateKeeperAddrCount; ++i) {
|
|
|
|
// Do we need to lookup the address?
|
|
|
|
const char* name = gateKeeperAddrList[i];
|
|
|
|
while (unsigned ch = *name) {
|
|
|
|
++name;
|
|
|
|
if (!(isdigit(ch) || ch == L'.' || ch == L':')) {
|
|
|
|
AsyncCancelId cancelId;
|
|
|
|
AsyncAddressLookupName(
|
|
|
|
&cancelId,
|
|
|
|
AsyncLookupCallback,
|
|
|
|
gateKeeperAddrList[i],
|
|
|
|
kNetDefaultClientPort,
|
|
|
|
nil
|
|
|
|
);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (!name[0]) {
|
|
|
|
plNetAddress addr(gateKeeperAddrList[i], kNetDefaultClientPort);
|
|
|
|
Connect(gateKeeperAddrList[i], addr);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
//============================================================================
|
|
|
|
void NetCliGateKeeperDisconnect () {
|
|
|
|
s_critsect.Enter();
|
|
|
|
{
|
|
|
|
while (CliGkConn * conn = s_conns.Head())
|
|
|
|
UnlinkAndAbandonConn_CS(conn);
|
|
|
|
s_active = nil;
|
|
|
|
}
|
|
|
|
s_critsect.Leave();
|
|
|
|
}
|
|
|
|
|
|
|
|
//============================================================================
|
|
|
|
void NetCliGateKeeperPingRequest (
|
|
|
|
unsigned pingTimeMs,
|
|
|
|
unsigned payloadBytes,
|
|
|
|
const void * payload,
|
|
|
|
FNetCliGateKeeperPingRequestCallback callback,
|
|
|
|
void * param
|
|
|
|
) {
|
|
|
|
PingRequestTrans * trans = new PingRequestTrans(
|
|
|
|
callback,
|
|
|
|
param,
|
|
|
|
pingTimeMs,
|
|
|
|
payloadBytes,
|
|
|
|
payload
|
|
|
|
);
|
|
|
|
NetTransSend(trans);
|
|
|
|
}
|
|
|
|
|
|
|
|
//============================================================================
|
|
|
|
void NetCliGateKeeperFileSrvIpAddressRequest (
|
|
|
|
FNetCliGateKeeperFileSrvIpAddressRequestCallback callback,
|
|
|
|
void * param,
|
|
|
|
bool isPatcher
|
|
|
|
) {
|
|
|
|
FileSrvIpAddressRequestTrans * trans = new FileSrvIpAddressRequestTrans(callback, param, isPatcher);
|
|
|
|
NetTransSend(trans);
|
|
|
|
}
|
|
|
|
|
|
|
|
//============================================================================
|
|
|
|
void NetCliGateKeeperAuthSrvIpAddressRequest (
|
|
|
|
FNetCliGateKeeperAuthSrvIpAddressRequestCallback callback,
|
|
|
|
void * param
|
|
|
|
) {
|
|
|
|
AuthSrvIpAddressRequestTrans * trans = new AuthSrvIpAddressRequestTrans(callback, param);
|
|
|
|
NetTransSend(trans);
|
|
|
|
}
|