2003-11-01 06:35:29 +01:00
|
|
|
/* -*-mode: C; fill-column: 78; c-basic-offset: 4; -*- */
|
|
|
|
/*
|
2007-12-05 07:33:37 +01:00
|
|
|
* Copyright 2001-2007 by Eric House (xwords@eehouse.org). All rights reserved.
|
2003-11-01 06:35:29 +01: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 2
|
|
|
|
* 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.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#ifdef USE_STDIO
|
|
|
|
# include <stdio.h>
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#include "comms.h"
|
|
|
|
|
|
|
|
#include "util.h"
|
2004-10-09 01:54:57 +02:00
|
|
|
#include "game.h"
|
2003-11-01 06:35:29 +01:00
|
|
|
#include "xwstream.h"
|
|
|
|
#include "memstream.h"
|
2005-03-19 23:01:38 +01:00
|
|
|
#include "xwrelay.h"
|
2005-03-06 18:56:34 +01:00
|
|
|
#include "strutils.h"
|
2003-11-01 06:35:29 +01:00
|
|
|
|
|
|
|
#define cEND 0x65454e44
|
2005-06-23 06:16:53 +02:00
|
|
|
#define HEARTBEAT_NONE 0
|
2003-11-01 06:35:29 +01:00
|
|
|
|
|
|
|
#ifndef XWFEATURE_STANDALONE_ONLY
|
|
|
|
|
2007-11-19 00:43:27 +01:00
|
|
|
#if defined RELAY_HEARTBEAT && defined COMMS_HEARTBEAT
|
|
|
|
compilation_error_here( "Choose one or the other or none." );
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#ifdef COMMS_HEARTBEAT
|
|
|
|
/* It might make sense for this to be a parameter or somehow tied to the
|
|
|
|
platform and transport. But in that case it'd have to be passed across
|
|
|
|
since all devices must agree. */
|
2007-12-05 07:33:37 +01:00
|
|
|
# ifndef HB_INTERVAL
|
|
|
|
# define HB_INTERVAL 5
|
|
|
|
# endif
|
2007-11-19 00:43:27 +01:00
|
|
|
#endif
|
|
|
|
|
2005-01-31 04:31:50 +01:00
|
|
|
EXTERN_C_START
|
|
|
|
|
2003-11-01 06:35:29 +01:00
|
|
|
typedef struct MsgQueueElem {
|
|
|
|
struct MsgQueueElem* next;
|
|
|
|
XP_U8* msg;
|
|
|
|
XP_U16 len;
|
2007-12-05 07:33:37 +01:00
|
|
|
XP_PlayerAddr channelNo;
|
2007-11-26 03:58:25 +01:00
|
|
|
XP_U16 sendCount; /* how many times sent? */
|
2005-03-19 23:01:38 +01:00
|
|
|
MsgID msgID; /* saved for ease of deletion */
|
2003-11-01 06:35:29 +01:00
|
|
|
} MsgQueueElem;
|
|
|
|
|
|
|
|
typedef struct AddressRecord {
|
|
|
|
struct AddressRecord* next;
|
|
|
|
CommsAddrRec addr;
|
2007-11-26 03:58:25 +01:00
|
|
|
#ifdef DEBUG
|
2007-12-05 07:33:37 +01:00
|
|
|
XP_U16 lastACK;
|
2003-11-01 06:35:29 +01:00
|
|
|
XP_U16 nUniqueBytes;
|
|
|
|
#endif
|
2005-03-19 23:01:38 +01:00
|
|
|
MsgID nextMsgID; /* on a per-channel basis */
|
2007-12-05 07:33:37 +01:00
|
|
|
MsgID lastMsgRcd; /* on a per-channel basis */
|
2007-11-19 00:43:27 +01:00
|
|
|
/* only used if COMMS_HEARTBEAT set except for serialization (to_stream) */
|
2003-11-01 06:35:29 +01:00
|
|
|
XP_PlayerAddr channelNo;
|
2006-10-02 16:26:56 +02:00
|
|
|
struct {
|
|
|
|
XWHostID hostID; /* used for relay case */
|
|
|
|
} r;
|
2007-12-05 07:33:37 +01:00
|
|
|
#ifdef COMMS_HEARTBEAT
|
|
|
|
XP_Bool initialSeen;
|
|
|
|
#endif
|
2003-11-01 06:35:29 +01:00
|
|
|
} AddressRecord;
|
|
|
|
|
2004-10-09 01:54:57 +02:00
|
|
|
#define ADDRESSRECORD_SIZE_68K 20
|
|
|
|
|
2005-03-19 23:01:38 +01:00
|
|
|
typedef enum {
|
|
|
|
COMMS_RELAYSTATE_UNCONNECTED
|
|
|
|
, COMMS_RELAYSTATE_CONNECT_PENDING
|
|
|
|
, COMMS_RELAYSTATE_CONNECTED
|
2005-10-02 17:39:38 +02:00
|
|
|
, COMMS_RELAYSTATE_ALLCONNECTED
|
2006-10-07 05:37:40 +02:00
|
|
|
} CommsRelayState;
|
2005-03-19 23:01:38 +01:00
|
|
|
|
2003-11-01 06:35:29 +01:00
|
|
|
struct CommsCtxt {
|
|
|
|
XW_UtilCtxt* util;
|
|
|
|
|
|
|
|
XP_U32 connID; /* 0 means ignore; otherwise must match */
|
2007-12-05 07:33:37 +01:00
|
|
|
XP_PlayerAddr nextChannelNo;
|
2005-03-19 23:01:38 +01:00
|
|
|
|
2003-11-01 06:35:29 +01:00
|
|
|
AddressRecord* recs; /* return addresses */
|
|
|
|
|
|
|
|
TransportSend sendproc;
|
2007-11-19 00:43:27 +01:00
|
|
|
#ifdef COMMS_HEARTBEAT
|
|
|
|
TransportReset resetproc;
|
2007-12-05 07:33:37 +01:00
|
|
|
XP_U32 lastMsgRcd;
|
2007-11-19 00:43:27 +01:00
|
|
|
#endif
|
2003-11-01 06:35:29 +01:00
|
|
|
void* sendClosure;
|
|
|
|
|
|
|
|
MsgQueueElem* msgQueueHead;
|
|
|
|
MsgQueueElem* msgQueueTail;
|
|
|
|
XP_U16 queueLen;
|
|
|
|
|
2007-11-19 00:43:27 +01:00
|
|
|
#ifdef COMMS_HEARTBEAT
|
|
|
|
XP_Bool doHeartbeat;
|
2007-12-05 07:33:37 +01:00
|
|
|
XP_Bool hbTimerPending;
|
|
|
|
XP_U32 lastMsgRcvdTime;
|
2007-11-19 00:43:27 +01:00
|
|
|
#endif
|
|
|
|
|
2006-10-10 03:34:37 +02:00
|
|
|
/* The following fields, down to isServer, are only used if
|
|
|
|
XWFEATURE_RELAY is defined, but I'm leaving them in here so apps built
|
|
|
|
both ways can open each other's saved games files.*/
|
2003-11-01 06:35:29 +01:00
|
|
|
CommsAddrRec addr;
|
|
|
|
|
2005-03-19 23:01:38 +01:00
|
|
|
/* Stuff for relays */
|
2006-10-02 16:26:56 +02:00
|
|
|
struct {
|
|
|
|
XWHostID myHostID; /* 0 if unset, 1 if acting as server,
|
|
|
|
random for client */
|
2006-10-07 05:37:40 +02:00
|
|
|
CommsRelayState relayState; /* not saved: starts at UNCONNECTED */
|
2006-10-02 16:26:56 +02:00
|
|
|
CookieID cookieID; /* not saved; temp standin for cookie; set
|
|
|
|
by relay */
|
|
|
|
/* permanent globally unique name, set by relay and forever after
|
|
|
|
associated with this game. Used to reconnect. */
|
|
|
|
XP_UCHAR connName[MAX_CONNNAME_LEN+1];
|
|
|
|
|
|
|
|
/* heartbeat: for periodic pings if relay thinks the network the
|
|
|
|
device is on requires them. Not saved since only valid when
|
|
|
|
connected, and we reconnect for every game and after restarting. */
|
|
|
|
XP_U16 heartbeat;
|
|
|
|
XP_U16 nPlayersHere;
|
|
|
|
XP_U16 nPlayersTotal;
|
|
|
|
XP_Bool connecting;
|
|
|
|
} r;
|
2006-04-07 05:08:23 +02:00
|
|
|
|
2003-11-01 06:35:29 +01:00
|
|
|
XP_Bool isServer;
|
|
|
|
#ifdef DEBUG
|
|
|
|
XP_U16 nUniqueBytes;
|
|
|
|
#endif
|
|
|
|
MPSLOT
|
|
|
|
};
|
|
|
|
|
2007-11-26 03:58:25 +01:00
|
|
|
#if defined XWFEATURE_BLUETOOTH || defined XWFEATURE_IP_DIRECT
|
2007-02-08 03:53:10 +01:00
|
|
|
typedef enum {
|
2007-11-26 03:58:25 +01:00
|
|
|
BTIPMSG_NONE = 0
|
|
|
|
,BTIPMSG_DATA
|
|
|
|
,BTIPMSG_RESET
|
2007-12-05 07:33:37 +01:00
|
|
|
,BTIPMSG_HB
|
2007-11-26 03:58:25 +01:00
|
|
|
} BTIPMsgType;
|
2007-02-08 03:53:10 +01:00
|
|
|
#endif
|
|
|
|
|
2003-11-01 06:35:29 +01:00
|
|
|
/****************************************************************************
|
|
|
|
* prototypes
|
|
|
|
****************************************************************************/
|
|
|
|
static AddressRecord* rememberChannelAddress( CommsCtxt* comms,
|
|
|
|
XP_PlayerAddr channelNo,
|
2006-09-14 03:25:40 +02:00
|
|
|
XWHostID id,
|
|
|
|
const CommsAddrRec* addr );
|
2007-12-05 07:33:37 +01:00
|
|
|
static void updateChannelAddress( AddressRecord* rec, const CommsAddrRec* addr );
|
2003-11-01 06:35:29 +01:00
|
|
|
static XP_Bool channelToAddress( CommsCtxt* comms, XP_PlayerAddr channelNo,
|
2007-02-08 03:53:10 +01:00
|
|
|
const CommsAddrRec** addr );
|
2008-01-07 02:10:43 +01:00
|
|
|
static AddressRecord* getRecordFor( CommsCtxt* comms, const CommsAddrRec* addr,
|
2003-11-01 06:35:29 +01:00
|
|
|
XP_PlayerAddr channelNo );
|
|
|
|
static XP_S16 sendMsg( CommsCtxt* comms, MsgQueueElem* elem );
|
|
|
|
static void addToQueue( CommsCtxt* comms, MsgQueueElem* newMsgElem );
|
2006-10-15 15:53:17 +02:00
|
|
|
static XP_U16 countAddrRecs( const CommsCtxt* comms );
|
2007-11-26 03:58:25 +01:00
|
|
|
static void sendConnect( CommsCtxt* comms );
|
|
|
|
|
2006-10-10 03:34:37 +02:00
|
|
|
#ifdef XWFEATURE_RELAY
|
2005-06-23 06:16:53 +02:00
|
|
|
static void relayConnect( CommsCtxt* comms );
|
2005-09-05 17:33:51 +02:00
|
|
|
static void relayDisconnect( CommsCtxt* comms );
|
2005-06-23 06:16:53 +02:00
|
|
|
static XP_Bool send_via_relay( CommsCtxt* comms, XWRELAY_Cmd cmd,
|
|
|
|
XWHostID destID, void* data, int dlen );
|
2005-03-19 23:01:38 +01:00
|
|
|
static XWHostID getDestID( CommsCtxt* comms, XP_PlayerAddr channelNo );
|
2007-11-19 00:43:27 +01:00
|
|
|
#endif
|
2007-12-05 07:33:37 +01:00
|
|
|
#if defined RELAY_HEARTBEAT || defined COMMS_HEARTBEAT
|
2005-06-23 06:16:53 +02:00
|
|
|
static void setHeartbeatTimer( CommsCtxt* comms );
|
2007-11-26 03:58:25 +01:00
|
|
|
#else
|
|
|
|
# define setHeartbeatTimer( comms )
|
2006-10-10 03:34:37 +02:00
|
|
|
#endif
|
2007-11-26 03:58:25 +01:00
|
|
|
#if defined XWFEATURE_BLUETOOTH || defined XWFEATURE_IP_DIRECT
|
|
|
|
static XP_S16 send_via_bt_or_ip( CommsCtxt* comms, BTIPMsgType typ,
|
|
|
|
XP_PlayerAddr channelNo,
|
|
|
|
void* data, int dlen );
|
2006-06-16 03:46:47 +02:00
|
|
|
#endif
|
2003-11-01 06:35:29 +01:00
|
|
|
|
|
|
|
/****************************************************************************
|
|
|
|
* implementation
|
|
|
|
****************************************************************************/
|
|
|
|
CommsCtxt*
|
|
|
|
comms_make( MPFORMAL XW_UtilCtxt* util, XP_Bool isServer,
|
2006-10-10 03:34:37 +02:00
|
|
|
XP_U16 XP_UNUSED_RELAY(nPlayersHere),
|
|
|
|
XP_U16 XP_UNUSED_RELAY(nPlayersTotal),
|
2007-11-19 00:43:27 +01:00
|
|
|
TransportSend sendproc, IF_CH(TransportReset resetproc)
|
|
|
|
void* closure )
|
2003-11-01 06:35:29 +01:00
|
|
|
{
|
|
|
|
CommsCtxt* result = (CommsCtxt*)XP_MALLOC( mpool, sizeof(*result) );
|
|
|
|
XP_MEMSET( result, 0, sizeof(*result) );
|
|
|
|
|
|
|
|
MPASSIGN(result->mpool, mpool);
|
|
|
|
|
|
|
|
result->isServer = isServer;
|
|
|
|
result->sendproc = sendproc;
|
2007-11-19 00:43:27 +01:00
|
|
|
#ifdef COMMS_HEARTBEAT
|
|
|
|
result->resetproc = resetproc;
|
|
|
|
#endif
|
2003-11-01 06:35:29 +01:00
|
|
|
result->sendClosure = closure;
|
|
|
|
result->util = util;
|
|
|
|
|
2006-10-10 03:34:37 +02:00
|
|
|
#ifdef XWFEATURE_RELAY
|
2006-10-02 16:26:56 +02:00
|
|
|
result->r.myHostID = isServer? HOST_ID_SERVER: HOST_ID_NONE;
|
|
|
|
XP_LOGF( "set myHostID to %d", result->r.myHostID );
|
2005-03-19 23:01:38 +01:00
|
|
|
|
2006-10-02 16:26:56 +02:00
|
|
|
result->r.relayState = COMMS_RELAYSTATE_UNCONNECTED;
|
|
|
|
result->r.nPlayersHere = nPlayersHere;
|
|
|
|
result->r.nPlayersTotal = nPlayersTotal;
|
2006-04-07 05:08:23 +02:00
|
|
|
#endif
|
2003-11-01 06:35:29 +01:00
|
|
|
return result;
|
|
|
|
} /* comms_make */
|
|
|
|
|
|
|
|
static void
|
|
|
|
cleanupInternal( CommsCtxt* comms )
|
|
|
|
{
|
|
|
|
MsgQueueElem* msg;
|
|
|
|
MsgQueueElem* next;
|
|
|
|
|
|
|
|
for ( msg = comms->msgQueueHead; !!msg; msg = next ) {
|
2005-01-31 04:31:50 +01:00
|
|
|
next = msg->next;
|
|
|
|
XP_FREE( comms->mpool, msg->msg );
|
|
|
|
XP_FREE( comms->mpool, msg );
|
2003-11-01 06:35:29 +01:00
|
|
|
}
|
|
|
|
comms->queueLen = 0;
|
|
|
|
comms->msgQueueHead = comms->msgQueueTail = (MsgQueueElem*)NULL;
|
|
|
|
} /* cleanupInternal */
|
|
|
|
|
|
|
|
static void
|
|
|
|
cleanupAddrRecs( CommsCtxt* comms )
|
|
|
|
{
|
|
|
|
AddressRecord* recs;
|
|
|
|
AddressRecord* next;
|
|
|
|
|
|
|
|
for ( recs = comms->recs; !!recs; recs = next ) {
|
|
|
|
next = recs->next;
|
|
|
|
XP_FREE( comms->mpool, recs );
|
|
|
|
}
|
|
|
|
comms->recs = (AddressRecord*)NULL;
|
|
|
|
} /* cleanupAddrRecs */
|
|
|
|
|
|
|
|
void
|
2005-10-02 17:39:38 +02:00
|
|
|
comms_reset( CommsCtxt* comms, XP_Bool isServer,
|
2006-10-10 03:34:37 +02:00
|
|
|
XP_U16 XP_UNUSED_RELAY(nPlayersHere),
|
|
|
|
XP_U16 XP_UNUSED_RELAY(nPlayersTotal) )
|
2003-11-01 06:35:29 +01:00
|
|
|
{
|
2007-02-06 06:49:45 +01:00
|
|
|
LOG_FUNC();
|
2006-10-10 03:34:37 +02:00
|
|
|
#ifdef XWFEATURE_RELAY
|
2005-09-05 17:33:51 +02:00
|
|
|
relayDisconnect( comms );
|
|
|
|
#endif
|
|
|
|
|
2003-11-01 06:35:29 +01:00
|
|
|
cleanupInternal( comms );
|
|
|
|
comms->isServer = isServer;
|
|
|
|
|
|
|
|
cleanupAddrRecs( comms );
|
|
|
|
|
|
|
|
comms->nextChannelNo = 0;
|
|
|
|
|
|
|
|
comms->connID = CONN_ID_NONE;
|
2006-10-10 03:34:37 +02:00
|
|
|
#ifdef XWFEATURE_RELAY
|
2006-10-02 16:26:56 +02:00
|
|
|
comms->r.cookieID = COOKIE_ID_NONE;
|
|
|
|
comms->r.nPlayersHere = nPlayersHere;
|
|
|
|
comms->r.nPlayersTotal = nPlayersTotal;
|
2005-09-05 17:33:51 +02:00
|
|
|
relayConnect( comms );
|
2005-09-04 22:30:47 +02:00
|
|
|
#endif
|
2007-02-06 06:49:45 +01:00
|
|
|
LOG_RETURN_VOID();
|
2003-11-01 06:35:29 +01:00
|
|
|
} /* comms_reset */
|
|
|
|
|
|
|
|
void
|
|
|
|
comms_destroy( CommsCtxt* comms )
|
|
|
|
{
|
2007-12-07 08:50:19 +01:00
|
|
|
CommsAddrRec aNew;
|
|
|
|
aNew.conType = COMMS_CONN_NONE;
|
|
|
|
util_addrChange( comms->util, &comms->addr, &aNew );
|
|
|
|
|
2003-11-01 06:35:29 +01:00
|
|
|
cleanupInternal( comms );
|
|
|
|
cleanupAddrRecs( comms );
|
|
|
|
|
|
|
|
XP_FREE( comms->mpool, comms );
|
|
|
|
} /* comms_destroy */
|
|
|
|
|
|
|
|
void
|
|
|
|
comms_setConnID( CommsCtxt* comms, XP_U32 connID )
|
|
|
|
{
|
|
|
|
comms->connID = connID;
|
2007-12-05 07:33:37 +01:00
|
|
|
XP_STATUSF( "%s: set connID to %lx", __func__, connID );
|
2003-11-01 06:35:29 +01:00
|
|
|
} /* comms_setConnID */
|
|
|
|
|
2005-03-19 23:01:38 +01:00
|
|
|
static void
|
|
|
|
addrFromStream( CommsAddrRec* addrP, XWStreamCtxt* stream )
|
|
|
|
{
|
|
|
|
CommsAddrRec addr;
|
|
|
|
|
2006-10-05 03:17:03 +02:00
|
|
|
addr.conType = stream_getU8( stream );
|
2005-03-19 23:01:38 +01:00
|
|
|
|
|
|
|
switch( addr.conType ) {
|
2007-12-07 08:50:19 +01:00
|
|
|
case COMMS_CONN_NONE:
|
2005-03-19 23:01:38 +01:00
|
|
|
break;
|
|
|
|
case COMMS_CONN_BT:
|
2006-09-08 09:14:24 +02:00
|
|
|
stringFromStreamHere( stream, addr.u.bt.hostName,
|
|
|
|
sizeof(addr.u.bt.hostName) );
|
2006-09-23 18:04:53 +02:00
|
|
|
stream_getBytes( stream, &addr.u.bt.btAddr.bits,
|
|
|
|
sizeof(addr.u.bt.btAddr.bits) );
|
2006-09-08 09:14:24 +02:00
|
|
|
break;
|
2005-03-19 23:01:38 +01:00
|
|
|
case COMMS_CONN_IR:
|
|
|
|
/* nothing to save */
|
|
|
|
break;
|
2007-11-26 03:58:25 +01:00
|
|
|
case COMMS_CONN_IP_DIRECT:
|
2005-03-19 23:01:38 +01:00
|
|
|
stringFromStreamHere( stream, addr.u.ip.hostName_ip,
|
|
|
|
sizeof(addr.u.ip.hostName_ip) );
|
|
|
|
addr.u.ip.ipAddr_ip = stream_getU32( stream );
|
|
|
|
addr.u.ip.port_ip = stream_getU16( stream );
|
|
|
|
break;
|
|
|
|
case COMMS_CONN_RELAY:
|
|
|
|
stringFromStreamHere( stream, addr.u.ip_relay.cookie,
|
|
|
|
sizeof(addr.u.ip_relay.cookie) );
|
|
|
|
stringFromStreamHere( stream, addr.u.ip_relay.hostName,
|
|
|
|
sizeof(addr.u.ip_relay.hostName) );
|
|
|
|
addr.u.ip_relay.ipAddr = stream_getU32( stream );
|
|
|
|
addr.u.ip_relay.port = stream_getU16( stream );
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
/* shut up, compiler */
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
XP_MEMCPY( addrP, &addr, sizeof(*addrP) );
|
|
|
|
} /* addrFromStream */
|
|
|
|
|
2003-11-01 06:35:29 +01:00
|
|
|
CommsCtxt*
|
|
|
|
comms_makeFromStream( MPFORMAL XWStreamCtxt* stream, XW_UtilCtxt* util,
|
2007-11-19 00:43:27 +01:00
|
|
|
TransportSend sendproc,
|
|
|
|
IF_CH(TransportReset resetproc ) void* closure )
|
2003-11-01 06:35:29 +01:00
|
|
|
{
|
|
|
|
CommsCtxt* comms;
|
|
|
|
XP_Bool isServer;
|
2005-10-02 17:39:38 +02:00
|
|
|
XP_U16 nAddrRecs, nPlayersHere, nPlayersTotal;
|
2003-11-01 06:35:29 +01:00
|
|
|
AddressRecord** prevsAddrNext;
|
|
|
|
MsgQueueElem** prevsQueueNext;
|
2006-10-05 03:17:03 +02:00
|
|
|
XP_U16 version = stream_getVersion( stream );
|
|
|
|
CommsAddrRec addr;
|
2005-03-08 06:33:38 +01:00
|
|
|
short i;
|
2003-11-01 06:35:29 +01:00
|
|
|
|
|
|
|
isServer = stream_getU8( stream );
|
2006-10-05 03:17:03 +02:00
|
|
|
if ( version < STREAM_VERS_RELAY ) {
|
|
|
|
XP_MEMSET( &addr, 0, sizeof(addr) );
|
|
|
|
addr.conType = COMMS_CONN_IR; /* all there was back then */
|
|
|
|
} else {
|
|
|
|
addrFromStream( &addr, stream );
|
|
|
|
}
|
|
|
|
|
|
|
|
if ( addr.conType == COMMS_CONN_RELAY ) {
|
|
|
|
nPlayersHere = (XP_U16)stream_getBits( stream, 4 );
|
|
|
|
nPlayersTotal = (XP_U16)stream_getBits( stream, 4 );
|
|
|
|
} else {
|
|
|
|
nPlayersHere = 0;
|
|
|
|
nPlayersTotal = 0;
|
|
|
|
}
|
2005-10-02 17:39:38 +02:00
|
|
|
comms = comms_make( MPPARM(mpool) util, isServer,
|
|
|
|
nPlayersHere, nPlayersTotal,
|
2007-11-19 00:43:27 +01:00
|
|
|
sendproc, IF_CH(resetproc) closure );
|
2006-10-05 03:17:03 +02:00
|
|
|
XP_MEMCPY( &comms->addr, &addr, sizeof(comms->addr) );
|
2003-11-01 06:35:29 +01:00
|
|
|
|
|
|
|
comms->connID = stream_getU32( stream );
|
|
|
|
comms->nextChannelNo = stream_getU16( stream );
|
2006-10-05 03:17:03 +02:00
|
|
|
if ( addr.conType == COMMS_CONN_RELAY ) {
|
|
|
|
comms->r.myHostID = stream_getU8( stream );
|
|
|
|
stringFromStreamHere( stream, comms->r.connName,
|
|
|
|
sizeof(comms->r.connName) );
|
|
|
|
}
|
2005-03-19 23:01:38 +01:00
|
|
|
|
2003-11-01 06:35:29 +01:00
|
|
|
#ifdef DEBUG
|
|
|
|
comms->nUniqueBytes = stream_getU16( stream );
|
|
|
|
#endif
|
|
|
|
comms->queueLen = stream_getU8( stream );
|
|
|
|
|
|
|
|
nAddrRecs = stream_getU8( stream );
|
|
|
|
prevsAddrNext = &comms->recs;
|
|
|
|
for ( i = 0; i < nAddrRecs; ++i ) {
|
|
|
|
AddressRecord* rec = (AddressRecord*)XP_MALLOC( mpool, sizeof(*rec));
|
|
|
|
XP_MEMSET( rec, 0, sizeof(*rec) );
|
2004-10-02 05:47:24 +02:00
|
|
|
|
2006-10-05 03:17:03 +02:00
|
|
|
addrFromStream( &rec->addr, stream );
|
2004-10-09 01:54:57 +02:00
|
|
|
|
2003-11-01 06:35:29 +01:00
|
|
|
rec->nextMsgID = stream_getU16( stream );
|
2007-12-05 07:33:37 +01:00
|
|
|
rec->lastMsgRcd = stream_getU16( stream );
|
2003-11-01 06:35:29 +01:00
|
|
|
rec->channelNo = stream_getU16( stream );
|
2006-10-05 03:17:03 +02:00
|
|
|
if ( rec->addr.conType == COMMS_CONN_RELAY ) {
|
|
|
|
rec->r.hostID = stream_getU8( stream );
|
|
|
|
}
|
2003-11-01 06:35:29 +01:00
|
|
|
|
|
|
|
#ifdef DEBUG
|
|
|
|
rec->lastACK = stream_getU16( stream );
|
|
|
|
rec->nUniqueBytes = stream_getU16( stream );
|
|
|
|
#endif
|
|
|
|
|
|
|
|
*prevsAddrNext = rec;
|
|
|
|
prevsAddrNext = &rec->next;
|
|
|
|
}
|
|
|
|
|
|
|
|
prevsQueueNext = &comms->msgQueueHead;
|
|
|
|
for ( i = 0; i < comms->queueLen; ++i ) {
|
|
|
|
MsgQueueElem* msg = (MsgQueueElem*)XP_MALLOC( mpool, sizeof(*msg) );
|
|
|
|
|
|
|
|
msg->channelNo = stream_getU16( stream );
|
|
|
|
msg->msgID = stream_getU32( stream );
|
2007-11-26 03:58:25 +01:00
|
|
|
#ifdef COMMS_HEARTBEAT
|
|
|
|
msg->sendCount = 0;
|
|
|
|
#endif
|
2003-11-01 06:35:29 +01:00
|
|
|
msg->len = stream_getU16( stream );
|
|
|
|
msg->msg = (XP_U8*)XP_MALLOC( mpool, msg->len );
|
|
|
|
stream_getBytes( stream, msg->msg, msg->len );
|
|
|
|
|
|
|
|
msg->next = (MsgQueueElem*)NULL;
|
|
|
|
*prevsQueueNext = comms->msgQueueTail = msg;
|
|
|
|
comms->msgQueueTail = msg;
|
|
|
|
prevsQueueNext = &msg->next;
|
|
|
|
}
|
|
|
|
|
2004-10-02 05:47:24 +02:00
|
|
|
#ifdef DEBUG
|
2003-11-01 06:35:29 +01:00
|
|
|
XP_ASSERT( stream_getU32( stream ) == cEND );
|
2004-10-02 05:47:24 +02:00
|
|
|
#endif
|
2005-03-19 23:01:38 +01:00
|
|
|
|
2003-11-01 06:35:29 +01:00
|
|
|
return comms;
|
|
|
|
} /* comms_makeFromStream */
|
|
|
|
|
2005-04-03 19:00:59 +02:00
|
|
|
void
|
2005-07-06 02:58:20 +02:00
|
|
|
comms_start( CommsCtxt* comms )
|
2005-04-03 19:00:59 +02:00
|
|
|
{
|
2007-11-26 03:58:25 +01:00
|
|
|
#ifdef COMMS_HEARTBEAT
|
|
|
|
comms->doHeartbeat = comms->addr.conType != COMMS_CONN_IR;
|
|
|
|
#endif
|
|
|
|
|
|
|
|
sendConnect( comms );
|
|
|
|
} /* comms_start */
|
|
|
|
|
|
|
|
static void
|
|
|
|
sendConnect( CommsCtxt* comms )
|
|
|
|
{
|
|
|
|
switch( comms->addr.conType ) {
|
2006-10-10 03:34:37 +02:00
|
|
|
#ifdef XWFEATURE_RELAY
|
2007-11-26 03:58:25 +01:00
|
|
|
case COMMS_CONN_RELAY:
|
2006-10-02 16:26:56 +02:00
|
|
|
comms->r.relayState = COMMS_RELAYSTATE_UNCONNECTED;
|
2005-06-23 06:16:53 +02:00
|
|
|
relayConnect( comms );
|
2007-11-26 03:58:25 +01:00
|
|
|
break;
|
2006-10-10 03:34:37 +02:00
|
|
|
#endif
|
2007-11-26 03:58:25 +01:00
|
|
|
#if defined XWFEATURE_BLUETOOTH || defined XWFEATURE_IP_DIRECT
|
|
|
|
case COMMS_CONN_BT:
|
|
|
|
case COMMS_CONN_IP_DIRECT:
|
2007-12-05 07:33:37 +01:00
|
|
|
/* This will only work on host side when there's a single guest! */
|
|
|
|
(void)send_via_bt_or_ip( comms, BTIPMSG_RESET, CHANNEL_NONE, NULL, 0 );
|
2007-11-26 03:58:25 +01:00
|
|
|
(void)comms_resendAll( comms );
|
|
|
|
break;
|
2006-08-26 23:12:10 +02:00
|
|
|
#endif
|
2007-11-26 03:58:25 +01:00
|
|
|
default:
|
|
|
|
break;
|
2005-04-03 19:00:59 +02:00
|
|
|
}
|
2007-11-19 00:43:27 +01:00
|
|
|
|
2007-11-26 03:58:25 +01:00
|
|
|
setHeartbeatTimer( comms );
|
2006-10-10 03:34:37 +02:00
|
|
|
} /* comms_start */
|
2005-04-03 19:00:59 +02:00
|
|
|
|
2005-03-19 23:01:38 +01:00
|
|
|
static void
|
2006-10-15 15:53:17 +02:00
|
|
|
addrToStream( XWStreamCtxt* stream, const CommsAddrRec* addrP )
|
2005-03-19 23:01:38 +01:00
|
|
|
{
|
|
|
|
CommsAddrRec addr;
|
|
|
|
XP_MEMCPY( &addr, addrP, sizeof(addr) );
|
|
|
|
|
2006-10-05 03:17:03 +02:00
|
|
|
stream_putU8( stream, addr.conType );
|
2005-03-19 23:01:38 +01:00
|
|
|
|
|
|
|
switch( addr.conType ) {
|
2007-12-07 08:50:19 +01:00
|
|
|
case COMMS_CONN_NONE:
|
2007-12-09 02:59:15 +01:00
|
|
|
/* nothing to write */
|
2005-03-19 23:01:38 +01:00
|
|
|
break;
|
|
|
|
case COMMS_CONN_BT:
|
2006-09-08 09:14:24 +02:00
|
|
|
stringToStream( stream, addr.u.bt.hostName );
|
2006-09-23 18:04:53 +02:00
|
|
|
/* sizeof(.bits) below defeats ARM's padding. */
|
|
|
|
stream_putBytes( stream, &addr.u.bt.btAddr.bits,
|
|
|
|
sizeof(addr.u.bt.btAddr.bits) );
|
2006-09-08 09:14:24 +02:00
|
|
|
break;
|
2005-03-19 23:01:38 +01:00
|
|
|
case COMMS_CONN_IR:
|
|
|
|
/* nothing to save */
|
|
|
|
break;
|
2007-11-26 03:58:25 +01:00
|
|
|
case COMMS_CONN_IP_DIRECT:
|
2005-03-19 23:01:38 +01:00
|
|
|
stringToStream( stream, addr.u.ip.hostName_ip );
|
|
|
|
stream_putU32( stream, addr.u.ip.ipAddr_ip );
|
|
|
|
stream_putU16( stream, addr.u.ip.port_ip );
|
|
|
|
break;
|
|
|
|
case COMMS_CONN_RELAY:
|
|
|
|
stringToStream( stream, addr.u.ip_relay.cookie );
|
|
|
|
stringToStream( stream, addr.u.ip_relay.hostName );
|
|
|
|
stream_putU32( stream, addr.u.ip_relay.ipAddr );
|
|
|
|
stream_putU16( stream, addr.u.ip_relay.port );
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
} /* addrToStream */
|
|
|
|
|
2003-11-01 06:35:29 +01:00
|
|
|
void
|
2006-10-15 15:53:17 +02:00
|
|
|
comms_writeToStream( const CommsCtxt* comms, XWStreamCtxt* stream )
|
2003-11-01 06:35:29 +01:00
|
|
|
{
|
2005-03-08 06:33:38 +01:00
|
|
|
XP_U16 nAddrRecs;
|
2003-11-01 06:35:29 +01:00
|
|
|
AddressRecord* rec;
|
|
|
|
MsgQueueElem* msg;
|
|
|
|
|
|
|
|
stream_putU8( stream, (XP_U8)comms->isServer );
|
2006-10-05 03:17:03 +02:00
|
|
|
addrToStream( stream, &comms->addr );
|
|
|
|
if ( comms->addr.conType == COMMS_CONN_RELAY ) {
|
|
|
|
stream_putBits( stream, 4, comms->r.nPlayersHere );
|
|
|
|
stream_putBits( stream, 4, comms->r.nPlayersTotal );
|
|
|
|
}
|
2005-03-19 23:01:38 +01:00
|
|
|
|
2003-11-01 06:35:29 +01:00
|
|
|
stream_putU32( stream, comms->connID );
|
|
|
|
stream_putU16( stream, comms->nextChannelNo );
|
2006-10-05 03:17:03 +02:00
|
|
|
if ( comms->addr.conType == COMMS_CONN_RELAY ) {
|
|
|
|
stream_putU8( stream, comms->r.myHostID );
|
|
|
|
stringToStream( stream, comms->r.connName );
|
|
|
|
}
|
2005-03-19 23:01:38 +01:00
|
|
|
|
2003-11-01 06:35:29 +01:00
|
|
|
#ifdef DEBUG
|
|
|
|
stream_putU16( stream, comms->nUniqueBytes );
|
|
|
|
#endif
|
|
|
|
|
|
|
|
XP_ASSERT( comms->queueLen <= 255 );
|
|
|
|
stream_putU8( stream, (XP_U8)comms->queueLen );
|
|
|
|
|
|
|
|
nAddrRecs = countAddrRecs(comms);
|
|
|
|
stream_putU8( stream, (XP_U8)nAddrRecs );
|
|
|
|
|
|
|
|
for ( rec = comms->recs; !!rec; rec = rec->next ) {
|
2005-03-19 23:01:38 +01:00
|
|
|
|
2004-10-02 05:47:24 +02:00
|
|
|
CommsAddrRec* addr = &rec->addr;
|
2005-03-19 23:01:38 +01:00
|
|
|
addrToStream( stream, addr );
|
2004-10-02 05:47:24 +02:00
|
|
|
|
2003-11-01 06:35:29 +01:00
|
|
|
stream_putU16( stream, (XP_U16)rec->nextMsgID );
|
2007-12-05 07:33:37 +01:00
|
|
|
stream_putU16( stream, (XP_U16)rec->lastMsgRcd );
|
2003-11-01 06:35:29 +01:00
|
|
|
stream_putU16( stream, rec->channelNo );
|
2006-10-05 03:17:03 +02:00
|
|
|
if ( rec->addr.conType == COMMS_CONN_RELAY ) {
|
|
|
|
stream_putU8( stream, rec->r.hostID ); /* unneeded unless RELAY */
|
|
|
|
}
|
2003-11-01 06:35:29 +01:00
|
|
|
#ifdef DEBUG
|
|
|
|
stream_putU16( stream, rec->lastACK );
|
|
|
|
stream_putU16( stream, rec->nUniqueBytes );
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
|
|
|
for ( msg = comms->msgQueueHead; !!msg; msg = msg->next ) {
|
|
|
|
stream_putU16( stream, msg->channelNo );
|
|
|
|
stream_putU32( stream, msg->msgID );
|
|
|
|
|
|
|
|
stream_putU16( stream, msg->len );
|
|
|
|
stream_putBytes( stream, msg->msg, msg->len );
|
|
|
|
}
|
|
|
|
|
|
|
|
#ifdef DEBUG
|
|
|
|
stream_putU32( stream, cEND );
|
|
|
|
#endif
|
|
|
|
} /* comms_writeToStream */
|
|
|
|
|
2005-06-27 07:45:28 +02:00
|
|
|
void
|
2006-10-15 15:53:17 +02:00
|
|
|
comms_getAddr( const CommsCtxt* comms, CommsAddrRec* addr )
|
2003-11-01 06:35:29 +01:00
|
|
|
{
|
2005-04-03 04:49:04 +02:00
|
|
|
XP_ASSERT( !!comms );
|
2003-11-01 06:35:29 +01:00
|
|
|
XP_MEMCPY( addr, &comms->addr, sizeof(*addr) );
|
|
|
|
} /* comms_getAddr */
|
|
|
|
|
|
|
|
void
|
2005-07-23 17:28:15 +02:00
|
|
|
comms_setAddr( CommsCtxt* comms, const CommsAddrRec* addr )
|
2003-11-01 06:35:29 +01:00
|
|
|
{
|
2005-09-04 22:30:47 +02:00
|
|
|
XP_ASSERT( comms != NULL );
|
2007-11-26 03:58:25 +01:00
|
|
|
#if defined XWFEATURE_RELAY || defined XWFEATURE_BLUETOOTH || defined XWFEATURE_IP_DIRECT
|
2006-09-08 09:14:24 +02:00
|
|
|
util_addrChange( comms->util, &comms->addr, addr );
|
2003-11-01 06:35:29 +01:00
|
|
|
#endif
|
2005-04-03 19:00:59 +02:00
|
|
|
XP_MEMCPY( &comms->addr, addr, sizeof(comms->addr) );
|
2005-03-19 23:01:38 +01:00
|
|
|
|
2007-11-26 03:58:25 +01:00
|
|
|
#ifdef COMMS_HEARTBEAT
|
|
|
|
comms->doHeartbeat = comms->addr.conType != COMMS_CONN_IR;
|
2007-02-08 03:53:10 +01:00
|
|
|
#endif
|
2007-11-26 03:58:25 +01:00
|
|
|
sendConnect( comms );
|
|
|
|
|
2003-11-01 06:35:29 +01:00
|
|
|
} /* comms_setAddr */
|
|
|
|
|
2005-06-27 07:45:28 +02:00
|
|
|
void
|
|
|
|
comms_getInitialAddr( CommsAddrRec* addr )
|
|
|
|
{
|
2006-10-15 16:09:17 +02:00
|
|
|
#if defined XWFEATURE_RELAY
|
2006-09-15 09:33:59 +02:00
|
|
|
addr->conType = COMMS_CONN_RELAY; /* for temporary ease in debugging */
|
2006-08-26 23:12:10 +02:00
|
|
|
addr->u.ip_relay.ipAddr = 0L; /* force 'em to set it */
|
|
|
|
addr->u.ip_relay.port = 10999;
|
|
|
|
{
|
|
|
|
char* name = "eehouse.org";
|
|
|
|
XP_MEMCPY( addr->u.ip_relay.hostName, name, XP_STRLEN(name)+1 );
|
|
|
|
}
|
|
|
|
addr->u.ip_relay.cookie[0] = '\0';
|
2006-10-15 16:09:17 +02:00
|
|
|
#elif defined PLATFORM_PALM
|
|
|
|
/* default values; default is still IR where there's a choice, at least on
|
|
|
|
Palm... */
|
|
|
|
addr->conType = COMMS_CONN_IR;
|
|
|
|
#else
|
|
|
|
addr->conType = COMMS_CONN_BT;
|
2006-08-26 23:12:10 +02:00
|
|
|
#endif
|
2005-06-27 07:45:28 +02:00
|
|
|
} /* comms_getInitialAddr */
|
|
|
|
|
2008-01-07 02:10:43 +01:00
|
|
|
XP_Bool
|
|
|
|
comms_checkAddr( DeviceRole role, const CommsAddrRec* addr, XW_UtilCtxt* util )
|
|
|
|
{
|
|
|
|
XP_Bool ok = XP_TRUE;
|
|
|
|
/* make sure the user's given us enough information to make a connection */
|
|
|
|
if ( role == SERVER_ISCLIENT ) {
|
|
|
|
if ( addr->conType == COMMS_CONN_BT ) {
|
|
|
|
XP_U32 empty = 0L; /* check four bytes to save some code */
|
|
|
|
if ( !XP_MEMCMP( &empty, &addr->u.bt.btAddr, sizeof(empty) ) ) {
|
|
|
|
ok = XP_FALSE;
|
|
|
|
if ( !!util ) {
|
|
|
|
util_userError( util, STR_NEED_BT_HOST_ADDR );
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return ok;
|
|
|
|
} /* comms_checkAddr */
|
|
|
|
|
2005-06-27 07:45:28 +02:00
|
|
|
CommsConnType
|
2006-10-15 15:53:17 +02:00
|
|
|
comms_getConType( const CommsCtxt* comms )
|
2003-11-01 06:35:29 +01:00
|
|
|
{
|
2007-12-09 02:59:15 +01:00
|
|
|
XP_ASSERT( !!comms ); /* or: return COMMS_CONN_NONE */
|
2003-11-01 06:35:29 +01:00
|
|
|
return comms->addr.conType;
|
|
|
|
} /* comms_getConType */
|
|
|
|
|
2006-09-08 09:14:24 +02:00
|
|
|
XP_Bool
|
2006-10-15 15:53:17 +02:00
|
|
|
comms_getIsServer( const CommsCtxt* comms )
|
2006-09-08 09:14:24 +02:00
|
|
|
{
|
2007-09-15 15:46:59 +02:00
|
|
|
XP_ASSERT( !!comms );
|
2006-09-08 09:14:24 +02:00
|
|
|
return comms->isServer;
|
|
|
|
}
|
|
|
|
|
2007-11-26 03:58:25 +01:00
|
|
|
static MsgQueueElem*
|
|
|
|
makeElemWithID( CommsCtxt* comms, MsgID msgID, AddressRecord* rec,
|
|
|
|
XP_PlayerAddr channelNo, XWStreamCtxt* stream )
|
2003-11-01 06:35:29 +01:00
|
|
|
{
|
|
|
|
XP_U16 headerLen;
|
2007-11-19 00:43:27 +01:00
|
|
|
XP_U16 streamSize = NULL == stream? 0 : stream_getSize( stream );
|
2007-12-05 07:33:37 +01:00
|
|
|
MsgID lastMsgRcd = (!!rec)? rec->lastMsgRcd : 0;
|
2003-11-01 06:35:29 +01:00
|
|
|
MsgQueueElem* newMsgElem;
|
|
|
|
XWStreamCtxt* msgStream;
|
|
|
|
|
|
|
|
#ifdef DEBUG
|
|
|
|
if ( !!rec ) {
|
|
|
|
rec->nUniqueBytes += streamSize;
|
|
|
|
} else {
|
|
|
|
comms->nUniqueBytes += streamSize;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
newMsgElem = (MsgQueueElem*)XP_MALLOC( comms->mpool,
|
|
|
|
sizeof( *newMsgElem ) );
|
|
|
|
newMsgElem->channelNo = channelNo;
|
|
|
|
newMsgElem->msgID = msgID;
|
2007-11-26 03:58:25 +01:00
|
|
|
#ifdef COMMS_HEARTBEAT
|
|
|
|
newMsgElem->sendCount = 0;
|
|
|
|
#endif
|
2003-11-01 06:35:29 +01:00
|
|
|
|
|
|
|
msgStream = mem_stream_make( MPPARM(comms->mpool)
|
|
|
|
util_getVTManager(comms->util),
|
|
|
|
NULL, 0,
|
|
|
|
(MemStreamCloseCallback)NULL );
|
|
|
|
stream_open( msgStream );
|
2007-12-05 07:33:37 +01:00
|
|
|
XP_LOGF( "%s: putting connID %ld", __func__, comms->connID );
|
2005-03-06 18:56:34 +01:00
|
|
|
stream_putU32( msgStream, comms->connID );
|
|
|
|
|
2003-11-01 06:35:29 +01:00
|
|
|
stream_putU16( msgStream, channelNo );
|
|
|
|
stream_putU32( msgStream, msgID );
|
2007-12-05 07:33:37 +01:00
|
|
|
XP_LOGF( "put lastMsgRcd: %ld", lastMsgRcd );
|
2003-11-01 06:35:29 +01:00
|
|
|
stream_putU32( msgStream, lastMsgRcd );
|
|
|
|
|
|
|
|
headerLen = stream_getSize( msgStream );
|
|
|
|
newMsgElem->len = streamSize + headerLen;
|
|
|
|
newMsgElem->msg = (XP_U8*)XP_MALLOC( comms->mpool, newMsgElem->len );
|
|
|
|
|
|
|
|
stream_getBytes( msgStream, newMsgElem->msg, headerLen );
|
|
|
|
stream_destroy( msgStream );
|
|
|
|
|
2007-11-19 00:43:27 +01:00
|
|
|
if ( 0 < streamSize ) {
|
|
|
|
stream_getBytes( stream, newMsgElem->msg + headerLen, streamSize );
|
|
|
|
}
|
2003-11-01 06:35:29 +01:00
|
|
|
|
2007-11-26 03:58:25 +01:00
|
|
|
return newMsgElem;
|
|
|
|
} /* makeElemWithID */
|
2007-11-19 00:43:27 +01:00
|
|
|
|
|
|
|
/* Send a message using the sequentially next MsgID. Save the message so
|
|
|
|
* resend can work. */
|
|
|
|
XP_S16
|
|
|
|
comms_send( CommsCtxt* comms, XWStreamCtxt* stream )
|
|
|
|
{
|
|
|
|
XP_PlayerAddr channelNo = stream_getAddress( stream );
|
2008-01-07 02:10:43 +01:00
|
|
|
AddressRecord* rec = getRecordFor( comms, NULL, channelNo );
|
2007-11-19 00:43:27 +01:00
|
|
|
MsgID msgID = (!!rec)? ++rec->nextMsgID : 0;
|
2007-11-26 03:58:25 +01:00
|
|
|
MsgQueueElem* elem;
|
|
|
|
XP_S16 result = -1;
|
2007-11-19 00:43:27 +01:00
|
|
|
|
2007-12-05 07:33:37 +01:00
|
|
|
|
|
|
|
XP_DEBUGF( "%s: assigning msgID=" XP_LD " on chnl %d", __func__,
|
|
|
|
msgID, channelNo );
|
2007-11-26 03:58:25 +01:00
|
|
|
|
|
|
|
elem = makeElemWithID( comms, msgID, rec, channelNo, stream );
|
|
|
|
if ( NULL != elem ) {
|
|
|
|
addToQueue( comms, elem );
|
|
|
|
result = sendMsg( comms, elem );
|
|
|
|
}
|
|
|
|
return result;
|
2003-11-01 06:35:29 +01:00
|
|
|
} /* comms_send */
|
|
|
|
|
|
|
|
/* Add new message to the end of the list. The list needs to be kept in order
|
|
|
|
* by ascending msgIDs within each channel since if there's a resend that's
|
|
|
|
* the order in which they need to be sent.
|
|
|
|
*/
|
|
|
|
static void
|
|
|
|
addToQueue( CommsCtxt* comms, MsgQueueElem* newMsgElem )
|
|
|
|
{
|
|
|
|
newMsgElem->next = (MsgQueueElem*)NULL;
|
|
|
|
if ( !comms->msgQueueHead ) {
|
|
|
|
comms->msgQueueHead = comms->msgQueueTail = newMsgElem;
|
|
|
|
XP_ASSERT( comms->queueLen == 0 );
|
|
|
|
comms->queueLen = 1;
|
|
|
|
} else {
|
|
|
|
XP_ASSERT( !!comms->msgQueueTail );
|
|
|
|
comms->msgQueueTail->next = newMsgElem;
|
|
|
|
comms->msgQueueTail = newMsgElem;
|
|
|
|
|
|
|
|
XP_ASSERT( comms->queueLen > 0 );
|
|
|
|
++comms->queueLen;
|
|
|
|
}
|
|
|
|
XP_STATUSF( "addToQueue: queueLen now %d", comms->queueLen );
|
|
|
|
} /* addToQueue */
|
|
|
|
|
|
|
|
#ifdef DEBUG
|
|
|
|
static void
|
|
|
|
printQueue( CommsCtxt* comms )
|
|
|
|
{
|
|
|
|
MsgQueueElem* elem;
|
|
|
|
short i;
|
|
|
|
|
|
|
|
for ( elem = comms->msgQueueHead, i = 0; i < comms->queueLen;
|
|
|
|
elem = elem->next, ++i ) {
|
2006-09-27 03:54:53 +02:00
|
|
|
XP_STATUSF( "\t%d: channel: %d; msgID=" XP_LD,
|
2003-11-01 06:35:29 +01:00
|
|
|
i+1, elem->channelNo, elem->msgID );
|
|
|
|
}
|
|
|
|
}
|
|
|
|
#endif
|
2006-09-17 07:06:46 +02:00
|
|
|
|
2007-11-26 03:58:25 +01:00
|
|
|
static void
|
|
|
|
freeElem( const CommsCtxt* comms, MsgQueueElem* elem )
|
|
|
|
{
|
|
|
|
XP_FREE( comms->mpool, elem->msg );
|
|
|
|
XP_FREE( comms->mpool, elem );
|
|
|
|
}
|
|
|
|
|
2003-11-01 06:35:29 +01:00
|
|
|
/* We've received on some channel a message with a certain ID. This means
|
|
|
|
* that all messages sent on that channel with lower IDs have been received
|
2007-02-06 06:49:45 +01:00
|
|
|
* and can be removed from our queue. BUT: if this ID is higher than any
|
|
|
|
* we've sent, don't remove. We may be starting a new game but have a server
|
|
|
|
* that's still on the old one.
|
2003-11-01 06:35:29 +01:00
|
|
|
*/
|
|
|
|
static void
|
|
|
|
removeFromQueue( CommsCtxt* comms, XP_PlayerAddr channelNo, MsgID msgID )
|
|
|
|
{
|
2007-02-06 06:49:45 +01:00
|
|
|
XP_STATUSF( "%s: remove msgs <= " XP_LD " for channel %d (queueLen: %d)",
|
|
|
|
__func__, msgID, channelNo, comms->queueLen );
|
|
|
|
|
2008-01-07 02:10:43 +01:00
|
|
|
if ( (channelNo == 0) || !!getRecordFor(comms, NULL, channelNo) ) {
|
2007-02-07 12:58:01 +01:00
|
|
|
MsgQueueElem dummy;
|
|
|
|
MsgQueueElem* keep = &dummy;
|
2007-02-06 06:49:45 +01:00
|
|
|
MsgQueueElem* elem;
|
|
|
|
MsgQueueElem* next;
|
|
|
|
|
|
|
|
for ( elem = comms->msgQueueHead; !!elem; elem = next ) {
|
2007-02-07 12:58:01 +01:00
|
|
|
XP_Bool knownGood = XP_FALSE;
|
2007-02-06 06:49:45 +01:00
|
|
|
next = elem->next;
|
|
|
|
|
|
|
|
/* remove the 0-channel message if we've established a channel
|
|
|
|
number. Only clients should have any 0-channel messages in the
|
|
|
|
queue, and receiving something from the server is an implicit
|
|
|
|
ACK -- IFF it isn't left over from the last game. */
|
|
|
|
|
|
|
|
if ( (elem->channelNo == 0) && (channelNo != 0) ) {
|
2007-02-07 12:58:01 +01:00
|
|
|
XP_ASSERT( !comms->isServer );
|
|
|
|
XP_ASSERT( elem->msgID == 0 );
|
2007-02-06 06:49:45 +01:00
|
|
|
} else if ( elem->channelNo != channelNo ) {
|
2007-02-07 12:58:01 +01:00
|
|
|
knownGood = XP_TRUE;
|
2007-02-06 06:49:45 +01:00
|
|
|
}
|
2006-09-15 09:33:59 +02:00
|
|
|
|
2007-02-07 12:58:01 +01:00
|
|
|
if ( !knownGood && (elem->msgID <= msgID) ) {
|
2007-11-26 03:58:25 +01:00
|
|
|
freeElem( comms, elem );
|
2007-02-06 06:49:45 +01:00
|
|
|
--comms->queueLen;
|
2006-09-27 03:54:53 +02:00
|
|
|
} else {
|
2007-02-07 12:58:01 +01:00
|
|
|
keep->next = elem;
|
2007-02-06 06:49:45 +01:00
|
|
|
keep = elem;
|
2006-09-15 09:33:59 +02:00
|
|
|
}
|
|
|
|
}
|
2007-02-07 12:58:01 +01:00
|
|
|
|
|
|
|
keep->next = NULL;
|
|
|
|
comms->msgQueueHead = dummy.next;
|
2003-11-01 06:35:29 +01:00
|
|
|
}
|
2006-09-27 03:54:53 +02:00
|
|
|
|
2007-02-06 06:49:45 +01:00
|
|
|
XP_STATUSF( "%s: queueLen now %d", __func__, comms->queueLen );
|
2003-11-01 06:35:29 +01:00
|
|
|
|
|
|
|
XP_ASSERT( comms->queueLen > 0 || comms->msgQueueHead == NULL );
|
|
|
|
|
2006-09-17 07:06:46 +02:00
|
|
|
#ifdef DEBUG
|
2003-11-01 06:35:29 +01:00
|
|
|
printQueue( comms );
|
2006-09-17 07:06:46 +02:00
|
|
|
#endif
|
2003-11-01 06:35:29 +01:00
|
|
|
} /* removeFromQueue */
|
|
|
|
|
|
|
|
static XP_S16
|
|
|
|
sendMsg( CommsCtxt* comms, MsgQueueElem* elem )
|
|
|
|
{
|
2007-11-26 03:58:25 +01:00
|
|
|
XP_S16 result = -1;
|
2003-11-01 06:35:29 +01:00
|
|
|
XP_PlayerAddr channelNo;
|
2007-02-08 16:17:23 +01:00
|
|
|
#if defined XWFEATURE_RELAY || defined XWFEATURE_BLUETOOTH
|
2007-02-08 03:53:10 +01:00
|
|
|
CommsConnType conType = comms_getConType( comms );
|
2007-02-08 16:17:23 +01:00
|
|
|
#endif
|
2003-11-01 06:35:29 +01:00
|
|
|
|
|
|
|
channelNo = elem->channelNo;
|
|
|
|
|
2005-07-23 17:28:15 +02:00
|
|
|
if ( 0 ) {
|
2006-10-10 03:34:37 +02:00
|
|
|
#ifdef XWFEATURE_RELAY
|
2007-02-08 03:53:10 +01:00
|
|
|
} else if ( conType == COMMS_CONN_RELAY ) {
|
2006-10-02 16:26:56 +02:00
|
|
|
if ( comms->r.relayState == COMMS_RELAYSTATE_ALLCONNECTED ) {
|
2005-03-19 23:01:38 +01:00
|
|
|
XWHostID destID = getDestID( comms, channelNo );
|
2005-06-23 06:16:53 +02:00
|
|
|
result = send_via_relay( comms, XWRELAY_MSG_TORELAY, destID,
|
|
|
|
elem->msg, elem->len );
|
2005-03-19 23:01:38 +01:00
|
|
|
} else {
|
2007-02-06 06:49:45 +01:00
|
|
|
XP_LOGF( "%s: skipping message: not connected", __func__ );
|
2005-03-19 23:01:38 +01:00
|
|
|
}
|
2007-02-08 03:53:10 +01:00
|
|
|
#endif
|
2007-11-26 03:58:25 +01:00
|
|
|
#if defined XWFEATURE_BLUETOOTH || defined XWFEATURE_IP_DIRECT
|
|
|
|
} else if ( conType == COMMS_CONN_BT || conType == COMMS_CONN_IP_DIRECT ) {
|
|
|
|
result = send_via_bt_or_ip( comms, BTIPMSG_DATA, channelNo,
|
|
|
|
elem->msg, elem->len );
|
|
|
|
#ifdef COMMS_HEARTBEAT
|
|
|
|
setHeartbeatTimer( comms );
|
|
|
|
#endif
|
2005-07-23 17:28:15 +02:00
|
|
|
#endif
|
2005-03-19 23:01:38 +01:00
|
|
|
} else {
|
2007-02-08 03:53:10 +01:00
|
|
|
const CommsAddrRec* addr;
|
|
|
|
(void)channelToAddress( comms, channelNo, &addr );
|
2005-03-19 23:01:38 +01:00
|
|
|
|
|
|
|
XP_ASSERT( !!comms->sendproc );
|
|
|
|
result = (*comms->sendproc)( elem->msg, elem->len, addr,
|
|
|
|
comms->sendClosure );
|
|
|
|
}
|
2007-11-26 03:58:25 +01:00
|
|
|
|
|
|
|
if ( result == elem->len ) {
|
|
|
|
++elem->sendCount;
|
|
|
|
XP_LOGF( "sendCount now %d", elem->sendCount );
|
|
|
|
}
|
|
|
|
|
2003-11-01 06:35:29 +01:00
|
|
|
return result;
|
|
|
|
} /* sendMsg */
|
|
|
|
|
|
|
|
XP_S16
|
|
|
|
comms_resendAll( CommsCtxt* comms )
|
|
|
|
{
|
|
|
|
MsgQueueElem* msg;
|
|
|
|
XP_S16 result = 0;
|
2007-03-19 00:38:53 +01:00
|
|
|
|
2006-11-11 23:37:36 +01:00
|
|
|
XP_ASSERT( !!comms );
|
|
|
|
|
2003-11-01 06:35:29 +01:00
|
|
|
for ( msg = comms->msgQueueHead; !!msg; msg = msg->next ) {
|
|
|
|
XP_S16 oneResult = sendMsg( comms, msg );
|
|
|
|
if ( result == 0 && oneResult != 0 ) {
|
|
|
|
result = oneResult;
|
|
|
|
}
|
2006-09-27 03:54:53 +02:00
|
|
|
XP_STATUSF( "resend: msgID=" XP_LD "; rslt=%d",
|
2006-09-08 09:14:24 +02:00
|
|
|
msg->msgID, oneResult );
|
2003-11-01 06:35:29 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
return result;
|
|
|
|
} /* comms_resend */
|
|
|
|
|
2006-10-10 03:34:37 +02:00
|
|
|
#ifdef XWFEATURE_RELAY
|
2005-03-19 23:01:38 +01:00
|
|
|
static XP_Bool
|
|
|
|
relayPreProcess( CommsCtxt* comms, XWStreamCtxt* stream, XWHostID* senderID )
|
|
|
|
{
|
2007-02-08 03:53:10 +01:00
|
|
|
XP_Bool consumed = XP_TRUE;
|
2005-03-19 23:01:38 +01:00
|
|
|
XWHostID destID, srcID;
|
2005-10-01 18:01:39 +02:00
|
|
|
CookieID cookieID;
|
2005-09-03 17:37:49 +02:00
|
|
|
XP_U8 relayErr;
|
2005-10-02 17:39:38 +02:00
|
|
|
XP_U8 hasName;
|
2005-03-19 23:01:38 +01:00
|
|
|
|
2005-10-30 17:14:13 +01:00
|
|
|
/* nothing for us to do here if not using relay */
|
2007-02-08 03:53:10 +01:00
|
|
|
XWRELAY_Cmd cmd = stream_getU8( stream );
|
|
|
|
switch( cmd ) {
|
|
|
|
|
|
|
|
case XWRELAY_CONNECT_RESP:
|
|
|
|
case XWRELAY_RECONNECT_RESP:
|
|
|
|
comms->r.relayState = COMMS_RELAYSTATE_CONNECTED;
|
|
|
|
comms->r.heartbeat = stream_getU16( stream );
|
|
|
|
comms->r.cookieID = stream_getU16( stream );
|
|
|
|
comms->r.myHostID = (XWHostID)stream_getU8( stream );
|
|
|
|
XP_LOGF( "got XWRELAY_CONNECTRESP; set cookieID = %d; "
|
|
|
|
"set hostid: %x",
|
|
|
|
comms->r.cookieID, comms->r.myHostID );
|
|
|
|
setHeartbeatTimer( comms );
|
|
|
|
break;
|
2005-10-02 17:39:38 +02:00
|
|
|
|
2007-02-08 03:53:10 +01:00
|
|
|
case XWRELAY_ALLHERE:
|
|
|
|
comms->r.relayState = COMMS_RELAYSTATE_ALLCONNECTED;
|
|
|
|
hasName = stream_getU8( stream );
|
|
|
|
if ( hasName ) {
|
|
|
|
stringFromStreamHere( stream, comms->r.connName,
|
|
|
|
sizeof(comms->r.connName) );
|
|
|
|
XP_LOGF( "read connName: %s", comms->r.connName );
|
|
|
|
} else {
|
|
|
|
XP_ASSERT( comms->r.connName[0] != '\0' );
|
|
|
|
}
|
2005-09-03 20:31:32 +02:00
|
|
|
|
2007-02-08 03:53:10 +01:00
|
|
|
/* We're [re-]connected now. Send any pending messages. This may
|
|
|
|
need to be done later since we're inside the platform's socket
|
|
|
|
read proc now. */
|
|
|
|
comms_resendAll( comms );
|
|
|
|
break;
|
|
|
|
case XWRELAY_MSG_FROMRELAY:
|
|
|
|
cookieID = stream_getU16( stream );
|
|
|
|
srcID = stream_getU8( stream );
|
|
|
|
destID = stream_getU8( stream );
|
|
|
|
XP_LOGF( "cookieID: %d; srcID: %x; destID: %x",
|
|
|
|
cookieID, srcID, destID );
|
|
|
|
/* If these values don't check out, drop it */
|
|
|
|
consumed = cookieID != comms->r.cookieID
|
|
|
|
|| destID != comms->r.myHostID;
|
|
|
|
if ( consumed ) {
|
|
|
|
XP_LOGF( "rejecting data message" );
|
|
|
|
} else {
|
|
|
|
*senderID = srcID;
|
|
|
|
}
|
|
|
|
break;
|
2005-09-03 08:57:01 +02:00
|
|
|
|
2007-02-08 03:53:10 +01:00
|
|
|
case XWRELAY_DISCONNECT_OTHER:
|
|
|
|
relayErr = stream_getU8( stream );
|
|
|
|
srcID = stream_getU8( stream );
|
|
|
|
XP_LOGF( "host id %x disconnected", srcID );
|
|
|
|
/* we will eventually want to tell the user which player's gone */
|
|
|
|
util_userError( comms->util, ERR_RELAY_BASE + relayErr );
|
|
|
|
break;
|
2005-09-03 17:37:49 +02:00
|
|
|
|
2007-02-08 03:53:10 +01:00
|
|
|
case XWRELAY_DISCONNECT_YOU: /* Close socket for this? */
|
|
|
|
case XWRELAY_CONNECTDENIED: /* Close socket for this? */
|
|
|
|
XP_LOGF( "XWRELAY_DISCONNECT_YOU|XWRELAY_CONNECTDENIED" );
|
|
|
|
relayErr = stream_getU8( stream );
|
|
|
|
util_userError( comms->util, ERR_RELAY_BASE + relayErr );
|
|
|
|
comms->r.relayState = COMMS_RELAYSTATE_UNCONNECTED;
|
|
|
|
/* fallthru */
|
|
|
|
default:
|
|
|
|
XP_LOGF( "dropping relay msg with cmd %d", (XP_U16)cmd );
|
|
|
|
}
|
2005-03-19 23:01:38 +01:00
|
|
|
|
2007-02-08 03:53:10 +01:00
|
|
|
return consumed;
|
|
|
|
} /* relayPreProcess */
|
|
|
|
#endif
|
|
|
|
|
2007-12-05 07:33:37 +01:00
|
|
|
#ifdef COMMS_HEARTBEAT
|
|
|
|
static void
|
|
|
|
noteHBReceived( CommsCtxt* comms/* , const CommsAddrRec* addr */ )
|
|
|
|
{
|
|
|
|
comms->lastMsgRcvdTime = util_getCurSeconds( comms->util );
|
|
|
|
setHeartbeatTimer( comms );
|
|
|
|
}
|
|
|
|
#else
|
|
|
|
# define noteHBReceived(a)
|
|
|
|
#endif
|
|
|
|
|
2007-11-26 03:58:25 +01:00
|
|
|
#if defined XWFEATURE_BLUETOOTH || defined XWFEATURE_IP_DIRECT
|
2007-02-08 03:53:10 +01:00
|
|
|
static XP_Bool
|
2007-11-26 03:58:25 +01:00
|
|
|
btIpPreProcess( CommsCtxt* comms, XWStreamCtxt* stream )
|
2007-02-08 03:53:10 +01:00
|
|
|
{
|
2007-11-26 03:58:25 +01:00
|
|
|
BTIPMsgType typ = (BTIPMsgType)stream_getU8( stream );
|
|
|
|
XP_Bool consumed = typ != BTIPMSG_DATA;
|
2007-02-08 03:53:10 +01:00
|
|
|
|
|
|
|
if ( consumed ) {
|
2007-11-26 03:58:25 +01:00
|
|
|
/* This is all there is so far */
|
2007-12-05 07:33:37 +01:00
|
|
|
if ( typ == BTIPMSG_RESET ) {
|
|
|
|
(void)comms_resendAll( comms );
|
|
|
|
} else if ( typ == BTIPMSG_HB ) {
|
|
|
|
/* noteHBReceived( comms, addr ); */
|
|
|
|
} else {
|
|
|
|
XP_ASSERT( 0 );
|
|
|
|
}
|
2005-03-19 23:01:38 +01:00
|
|
|
}
|
2007-02-08 03:53:10 +01:00
|
|
|
|
2005-03-19 23:01:38 +01:00
|
|
|
return consumed;
|
2007-11-26 03:58:25 +01:00
|
|
|
} /* btIpPreProcess */
|
2005-07-23 17:28:15 +02:00
|
|
|
#endif
|
2005-03-19 23:01:38 +01:00
|
|
|
|
2007-11-26 03:58:25 +01:00
|
|
|
static XP_Bool
|
|
|
|
preProcess( CommsCtxt* comms, XWStreamCtxt* stream,
|
2007-12-03 02:20:32 +01:00
|
|
|
XP_Bool* usingRelay, XWHostID* XP_UNUSED_RELAY(senderID) )
|
2007-11-26 03:58:25 +01:00
|
|
|
{
|
|
|
|
XP_Bool consumed = XP_FALSE;
|
|
|
|
switch ( comms->addr.conType ) {
|
|
|
|
#ifdef XWFEATURE_RELAY
|
|
|
|
/* relayPreProcess returns true if consumes the message. May just eat the
|
|
|
|
header and leave a regular message to be processed below. */
|
|
|
|
case COMMS_CONN_RELAY:
|
|
|
|
consumed = relayPreProcess( comms, stream, senderID );
|
|
|
|
if ( !consumed ) {
|
|
|
|
*usingRelay = comms->addr.conType == COMMS_CONN_RELAY;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
#endif
|
|
|
|
#if defined XWFEATURE_BLUETOOTH || defined XWFEATURE_IP_DIRECT
|
|
|
|
case COMMS_CONN_BT:
|
|
|
|
case COMMS_CONN_IP_DIRECT:
|
|
|
|
consumed = btIpPreProcess( comms, stream );
|
|
|
|
break;
|
|
|
|
#endif
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
return consumed;
|
|
|
|
} /* preProcess */
|
|
|
|
|
2007-12-05 07:33:37 +01:00
|
|
|
static AddressRecord*
|
2008-01-07 02:10:43 +01:00
|
|
|
getRecordFor( CommsCtxt* comms, const CommsAddrRec* addr,
|
|
|
|
XP_PlayerAddr channelNo )
|
2006-11-11 23:37:36 +01:00
|
|
|
{
|
2007-12-31 21:00:13 +01:00
|
|
|
CommsConnType conType;
|
2007-12-05 07:33:37 +01:00
|
|
|
AddressRecord* rec;
|
|
|
|
XP_Bool matched = XP_FALSE;
|
2007-12-31 21:00:13 +01:00
|
|
|
|
2008-01-07 02:10:43 +01:00
|
|
|
/* Use addr if we have it. Otherwise use channelNo if non-0 */
|
|
|
|
conType = !!addr? addr->conType : COMMS_CONN_NONE;
|
2007-12-31 21:00:13 +01:00
|
|
|
|
2007-12-05 07:33:37 +01:00
|
|
|
for ( rec = comms->recs; !!rec; rec = rec->next ) {
|
2008-01-07 02:10:43 +01:00
|
|
|
XP_ASSERT( !addr || (conType == rec->addr.conType) );
|
2007-12-05 07:33:37 +01:00
|
|
|
switch( conType ) {
|
|
|
|
case COMMS_CONN_RELAY:
|
|
|
|
if ( (addr->u.ip_relay.ipAddr == rec->addr.u.ip_relay.ipAddr)
|
|
|
|
&& (addr->u.ip_relay.port == rec->addr.u.ip_relay.port ) ) {
|
|
|
|
matched = XP_TRUE;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case COMMS_CONN_BT:
|
|
|
|
if ( 0 == XP_MEMCMP( &addr->u.bt.btAddr, &rec->addr.u.bt.btAddr,
|
|
|
|
sizeof(addr->u.bt.btAddr) ) ) {
|
|
|
|
matched = XP_TRUE;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case COMMS_CONN_IP_DIRECT:
|
|
|
|
if ( (addr->u.ip.ipAddr_ip == rec->addr.u.ip.ipAddr_ip)
|
|
|
|
&& (addr->u.ip.port_ip == rec->addr.u.ip.port_ip) ) {
|
|
|
|
matched = XP_TRUE;
|
2006-11-11 23:37:36 +01:00
|
|
|
}
|
2007-12-05 07:33:37 +01:00
|
|
|
break;
|
|
|
|
case COMMS_CONN_IR: /* no way to test */
|
2007-12-31 21:00:13 +01:00
|
|
|
break;
|
2008-01-07 02:10:43 +01:00
|
|
|
case COMMS_CONN_NONE:
|
|
|
|
matched = channelNo == rec->channelNo;
|
|
|
|
break;
|
2007-12-05 07:33:37 +01:00
|
|
|
default:
|
2007-12-31 21:00:13 +01:00
|
|
|
XP_ASSERT(0);
|
2007-12-05 07:33:37 +01:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
if ( matched ) {
|
|
|
|
break;
|
2006-11-11 23:37:36 +01:00
|
|
|
}
|
|
|
|
}
|
2007-12-05 07:33:37 +01:00
|
|
|
return rec;
|
|
|
|
} /* addrToRecord */
|
|
|
|
|
|
|
|
/* An initial message comes only from a client to a server, and from the
|
|
|
|
* server in response to that initial message. Once the inital messages are
|
|
|
|
* exchanged there's a connID associated. The greatest danger is that it's a
|
|
|
|
* dup, resent for whatever reason. To detect that we check that the address
|
|
|
|
* is unknown. But addresses can change, e.g. if a reset of a socket-based
|
|
|
|
* transport causes the local socket to change. How to deal with this?
|
|
|
|
* Likely a boolean set when we call comms->resetproc that causes us to accept
|
|
|
|
* changed addresses.
|
2003-11-01 06:35:29 +01:00
|
|
|
*
|
2007-12-05 07:33:37 +01:00
|
|
|
* But: before we're connected heartbeats will also come here, but with
|
|
|
|
* hasPayload false. We want to remember their address, but not give them a
|
|
|
|
* channel ID. So if we have a payload we insist that it's the first we've
|
|
|
|
* seen on this channel.
|
2003-11-01 06:35:29 +01:00
|
|
|
*
|
2007-12-05 07:33:37 +01:00
|
|
|
* If it's a HB, then we want to add a rec/channel if there's none, but mark
|
|
|
|
* it invalid
|
2003-11-01 06:35:29 +01:00
|
|
|
*/
|
2007-12-05 07:33:37 +01:00
|
|
|
static AddressRecord*
|
|
|
|
validateInitialMessage( CommsCtxt* comms, XP_Bool hasPayload,
|
|
|
|
const CommsAddrRec* addr, XWHostID senderID,
|
|
|
|
XP_PlayerAddr* channelNo )
|
|
|
|
{
|
|
|
|
#ifdef COMMS_HEARTBEAT
|
|
|
|
XP_Bool addRec = XP_FALSE;
|
2008-01-07 02:10:43 +01:00
|
|
|
AddressRecord* rec = getRecordFor( comms, addr, *channelNo );
|
2007-12-05 07:33:37 +01:00
|
|
|
LOG_FUNC();
|
|
|
|
|
|
|
|
if ( hasPayload ) {
|
|
|
|
if ( rec ) {
|
|
|
|
if ( rec->initialSeen ) {
|
|
|
|
rec = NULL; /* reject it! */
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
addRec = XP_TRUE;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
/* This is a heartbeat */
|
|
|
|
if ( !rec && comms->isServer ) {
|
|
|
|
addRec = XP_TRUE;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if ( addRec ) {
|
|
|
|
if ( comms->isServer ) {
|
|
|
|
XP_ASSERT( *channelNo == 0 );
|
|
|
|
*channelNo = ++comms->nextChannelNo;
|
|
|
|
}
|
|
|
|
rec = rememberChannelAddress( comms, *channelNo, senderID, addr );
|
|
|
|
if ( hasPayload ) {
|
|
|
|
rec->initialSeen = XP_TRUE;
|
|
|
|
} else {
|
|
|
|
rec = NULL;
|
|
|
|
}
|
|
|
|
}
|
2008-03-11 13:24:23 +01:00
|
|
|
LOG_RETURNF( XP_P, rec );
|
2007-12-05 07:33:37 +01:00
|
|
|
return rec;
|
|
|
|
#else
|
2008-01-07 02:10:43 +01:00
|
|
|
AddressRecord* rec = getRecordFor( comms, addr, *channelNo );
|
2007-12-05 07:33:37 +01:00
|
|
|
if ( !!rec ) {
|
|
|
|
rec = NULL; /* reject: we've already seen init message on channel */
|
|
|
|
} else {
|
|
|
|
if ( comms->isServer ) {
|
|
|
|
XP_ASSERT( *channelNo == 0 );
|
|
|
|
*channelNo = ++comms->nextChannelNo;
|
|
|
|
}
|
|
|
|
rec = rememberChannelAddress( comms, *channelNo, senderID, addr );
|
|
|
|
}
|
|
|
|
return rec;
|
|
|
|
#endif
|
|
|
|
} /* validateInitialMessage */
|
|
|
|
|
|
|
|
/* Messages with established connIDs are valid only if they have the msgID
|
|
|
|
* that's expected on that channel. Their addresses need to match what we
|
|
|
|
* have for that channel, and in fact we'll overwrite what we have in case a
|
|
|
|
* reset has changed the address. The danger is that somebody might sneak in
|
|
|
|
* with a forged message, but this isn't internet banking.
|
|
|
|
*/
|
|
|
|
static AddressRecord*
|
|
|
|
validateChannelMessage( CommsCtxt* comms, const CommsAddrRec* addr,
|
|
|
|
XP_PlayerAddr channelNo, MsgID msgID, MsgID lastMsgRcd )
|
|
|
|
|
|
|
|
{
|
|
|
|
AddressRecord* rec;
|
|
|
|
LOG_FUNC();
|
|
|
|
|
2008-01-07 02:10:43 +01:00
|
|
|
rec = getRecordFor( comms, NULL, channelNo );
|
2007-12-05 07:33:37 +01:00
|
|
|
if ( !!rec ) {
|
|
|
|
removeFromQueue( comms, channelNo, lastMsgRcd );
|
|
|
|
if ( msgID == rec->lastMsgRcd + 1 ) {
|
|
|
|
updateChannelAddress( rec, addr );
|
|
|
|
#ifdef DEBUG
|
|
|
|
rec->lastACK = (XP_U16)lastMsgRcd;
|
|
|
|
#endif
|
|
|
|
} else {
|
2008-03-11 13:24:23 +01:00
|
|
|
XP_LOGF( "%s: expected %ld, got %ld", __func__,
|
2007-12-05 07:33:37 +01:00
|
|
|
rec->lastMsgRcd + 1, msgID );
|
|
|
|
rec = NULL;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
XP_LOGF( "%s: no rec for addr", __func__ );
|
|
|
|
}
|
|
|
|
|
2008-03-11 13:24:23 +01:00
|
|
|
LOG_RETURNF( XP_P, rec );
|
2007-12-05 07:33:37 +01:00
|
|
|
return rec;
|
|
|
|
} /* validateChannelMessage */
|
|
|
|
|
2003-11-01 06:35:29 +01:00
|
|
|
XP_Bool
|
2005-03-09 16:18:17 +01:00
|
|
|
comms_checkIncomingStream( CommsCtxt* comms, XWStreamCtxt* stream,
|
2006-09-14 03:25:40 +02:00
|
|
|
const CommsAddrRec* addr )
|
2003-11-01 06:35:29 +01:00
|
|
|
{
|
2006-08-29 15:18:12 +02:00
|
|
|
XP_Bool validMessage = XP_FALSE;
|
2006-09-14 03:25:40 +02:00
|
|
|
XWHostID senderID = 0; /* unset; default for non-relay cases */
|
2005-10-01 18:01:39 +02:00
|
|
|
XP_Bool usingRelay = XP_FALSE;
|
2007-12-05 07:33:37 +01:00
|
|
|
AddressRecord* rec = NULL;
|
2005-03-06 18:56:34 +01:00
|
|
|
|
2005-03-20 16:01:59 +01:00
|
|
|
XP_ASSERT( addr == NULL || comms->addr.conType == addr->conType );
|
2005-03-19 23:01:38 +01:00
|
|
|
|
2007-11-26 03:58:25 +01:00
|
|
|
if ( !preProcess( comms, stream, &usingRelay, &senderID ) ) {
|
2007-12-05 07:33:37 +01:00
|
|
|
XP_U32 connID;
|
|
|
|
XP_PlayerAddr channelNo;
|
|
|
|
MsgID msgID;
|
|
|
|
MsgID lastMsgRcd;
|
|
|
|
|
|
|
|
/* reject too-small message */
|
|
|
|
if ( stream_getSize( stream ) >=
|
|
|
|
(sizeof(connID) + sizeof(channelNo)
|
|
|
|
+ sizeof(msgID) + sizeof(lastMsgRcd)) ) {
|
|
|
|
XP_U16 payloadSize;
|
|
|
|
|
2006-10-07 05:37:40 +02:00
|
|
|
connID = stream_getU32( stream );
|
2007-02-06 06:49:45 +01:00
|
|
|
XP_STATUSF( "%s: read connID of %lx", __func__, connID );
|
2007-12-05 07:33:37 +01:00
|
|
|
channelNo = stream_getU16( stream );
|
|
|
|
XP_STATUSF( "read channelNo %d", channelNo );
|
|
|
|
msgID = stream_getU32( stream );
|
|
|
|
lastMsgRcd = stream_getU32( stream );
|
|
|
|
XP_DEBUGF( "rcd: msgID=" XP_LD ",lastMsgRcd=" XP_LD " on chnl %d",
|
|
|
|
msgID, lastMsgRcd, channelNo );
|
|
|
|
|
|
|
|
payloadSize = stream_getSize( stream ) > 0; /* anything left? */
|
|
|
|
if ( connID == CONN_ID_NONE ) {
|
|
|
|
/* special case: initial message from client */
|
|
|
|
rec = validateInitialMessage( comms, payloadSize > 0, addr, senderID,
|
|
|
|
&channelNo );
|
|
|
|
} else if ( comms->connID == connID ) {
|
|
|
|
rec = validateChannelMessage( comms, addr, channelNo, msgID,
|
|
|
|
lastMsgRcd );
|
|
|
|
}
|
2006-10-07 05:37:40 +02:00
|
|
|
|
2007-12-05 07:33:37 +01:00
|
|
|
validMessage = NULL != rec;
|
|
|
|
if ( validMessage ) {
|
|
|
|
rec->lastMsgRcd = msgID;
|
|
|
|
XP_LOGF( "%s: set channel %d's lastMsgRcd to " XP_LD,
|
|
|
|
__func__, channelNo, msgID );
|
|
|
|
stream_setAddress( stream, channelNo );
|
|
|
|
validMessage = payloadSize > 0;
|
2003-11-01 06:35:29 +01:00
|
|
|
}
|
2006-08-29 15:18:12 +02:00
|
|
|
} else {
|
2007-12-02 20:13:25 +01:00
|
|
|
XP_LOGF( "%s: message too small", __func__ );
|
2003-11-01 06:35:29 +01:00
|
|
|
}
|
|
|
|
}
|
2007-12-05 07:33:37 +01:00
|
|
|
|
|
|
|
/* Call after we've had a chance to create rec for addr */
|
|
|
|
noteHBReceived( comms/* , addr */ );
|
|
|
|
|
2006-09-15 09:33:59 +02:00
|
|
|
LOG_RETURNF( "%d", (XP_U16)validMessage );
|
2003-11-01 06:35:29 +01:00
|
|
|
return validMessage;
|
2005-03-09 16:18:17 +01:00
|
|
|
} /* comms_checkIncomingStream */
|
2003-11-01 06:35:29 +01:00
|
|
|
|
2007-11-19 00:43:27 +01:00
|
|
|
#ifdef COMMS_HEARTBEAT
|
2007-12-05 07:33:37 +01:00
|
|
|
static void
|
|
|
|
sendEmptyMsg( CommsCtxt* comms, AddressRecord* rec )
|
|
|
|
{
|
|
|
|
MsgQueueElem* elem = makeElemWithID( comms,
|
|
|
|
0 /*rec? rec->lastMsgRcd : 0*/,
|
|
|
|
rec,
|
|
|
|
rec? rec->channelNo : 0, NULL );
|
|
|
|
sendMsg( comms, elem );
|
|
|
|
freeElem( comms, elem );
|
|
|
|
} /* sendEmptyMsg */
|
|
|
|
|
2007-11-19 00:43:27 +01:00
|
|
|
/* Heartbeat.
|
|
|
|
*
|
|
|
|
* Goal is to allow all participants to detect when another is gone quickly.
|
|
|
|
* Assumption is that transport is cheap: sending extra packets doesn't cost
|
|
|
|
* much money or bother (meaning: don't do this over IR! :-).
|
|
|
|
*
|
|
|
|
* Keep track of last time we heard from each channel and of when we last sent
|
|
|
|
* a packet. Run a timer, and when it fires: 1) check if we haven't heard
|
|
|
|
* since 2x the timer interval. If so, call alert function and reset the
|
|
|
|
* underlying (ip, bt) channel. If not, check how long since we last sent a
|
|
|
|
* packet on each channel. If it's been longer than since the last timer, and
|
|
|
|
* if there are not already packets in the queue on that channel, fire a HB
|
|
|
|
* packet.
|
|
|
|
*
|
|
|
|
* A HB packet is one whose msg ID is lower than the most recent ACK'd so that
|
|
|
|
* it's sure to be dropped on the other end and not to interfere with packets
|
|
|
|
* that might be resent.
|
|
|
|
*/
|
|
|
|
static void
|
|
|
|
heartbeat_checks( CommsCtxt* comms )
|
|
|
|
{
|
|
|
|
LOG_FUNC();
|
|
|
|
|
2007-12-05 07:33:37 +01:00
|
|
|
do {
|
|
|
|
if ( comms->lastMsgRcvdTime > 0 ) {
|
|
|
|
XP_U32 now = util_getCurSeconds( comms->util );
|
|
|
|
XP_U32 tooLongAgo = now - (HB_INTERVAL * 2);
|
|
|
|
if ( comms->lastMsgRcvdTime < tooLongAgo ) {
|
|
|
|
XP_LOGF( "calling reset proc; last was %ld secs too long ago",
|
|
|
|
tooLongAgo - comms->lastMsgRcvdTime );
|
|
|
|
(*comms->resetproc)(comms->sendClosure);
|
|
|
|
comms->lastMsgRcvdTime = 0;
|
|
|
|
break; /* outta here */
|
|
|
|
}
|
2007-11-26 03:58:25 +01:00
|
|
|
}
|
2007-11-19 00:43:27 +01:00
|
|
|
|
2007-12-05 07:33:37 +01:00
|
|
|
if ( comms->recs ) {
|
|
|
|
AddressRecord* rec;
|
|
|
|
for ( rec = comms->recs; !!rec; rec = rec->next ) {
|
|
|
|
sendEmptyMsg( comms, rec );
|
2007-11-26 03:58:25 +01:00
|
|
|
}
|
2007-12-05 07:33:37 +01:00
|
|
|
} else if ( !comms->isServer ) {
|
|
|
|
/* Client still waiting for inital ALL_REG message */
|
|
|
|
sendEmptyMsg( comms, NULL );
|
2007-11-19 00:43:27 +01:00
|
|
|
}
|
2007-12-05 07:33:37 +01:00
|
|
|
} while ( XP_FALSE );
|
2007-11-19 00:43:27 +01:00
|
|
|
|
2007-12-05 07:33:37 +01:00
|
|
|
setHeartbeatTimer( comms );
|
2007-11-19 00:43:27 +01:00
|
|
|
} /* heartbeat_checks */
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#if defined RELAY_HEARTBEAT || defined COMMS_HEARTBEAT
|
2008-04-12 17:36:31 +02:00
|
|
|
static XP_Bool
|
2006-08-16 15:44:44 +02:00
|
|
|
p_comms_timerFired( void* closure, XWTimerReason XP_UNUSED_DBG(why) )
|
2005-06-23 06:16:53 +02:00
|
|
|
{
|
|
|
|
CommsCtxt* comms = (CommsCtxt*)closure;
|
|
|
|
XP_ASSERT( why == TIMER_HEARTBEAT );
|
2007-12-05 07:33:37 +01:00
|
|
|
LOG_FUNC();
|
|
|
|
comms->hbTimerPending = XP_FALSE;
|
2007-11-19 00:43:27 +01:00
|
|
|
if (0 ) {
|
|
|
|
#ifdef RELAY_HEARTBEAT
|
|
|
|
} else if ( (comms->addr.conType == COMMS_CONN_RELAY )
|
|
|
|
&& (comms->r.heartbeat != HEARTBEAT_NONE) ) {
|
2005-06-23 06:16:53 +02:00
|
|
|
send_via_relay( comms, XWRELAY_HEARTBEAT, HOST_ID_NONE, NULL, 0 );
|
2005-07-05 22:57:37 +02:00
|
|
|
/* No need to reset timer. send_via_relay does that. */
|
2007-11-19 00:43:27 +01:00
|
|
|
#elif defined COMMS_HEARTBEAT
|
|
|
|
} else {
|
|
|
|
XP_ASSERT( comms->doHeartbeat );
|
|
|
|
heartbeat_checks( comms );
|
|
|
|
#endif
|
2005-06-23 06:16:53 +02:00
|
|
|
}
|
2008-04-12 17:36:31 +02:00
|
|
|
return XP_FALSE; /* no need for redraw */
|
2007-11-19 00:43:27 +01:00
|
|
|
} /* p_comms_timerFired */
|
2005-06-23 06:16:53 +02:00
|
|
|
|
|
|
|
static void
|
|
|
|
setHeartbeatTimer( CommsCtxt* comms )
|
|
|
|
{
|
2007-11-26 03:58:25 +01:00
|
|
|
LOG_FUNC();
|
2007-12-05 07:33:37 +01:00
|
|
|
if ( !comms->hbTimerPending ) {
|
|
|
|
XP_U16 when = 0;
|
2007-11-19 00:43:27 +01:00
|
|
|
#ifdef RELAY_HEARTBEAT
|
2007-12-05 07:33:37 +01:00
|
|
|
if ( comms->addr.conType == COMMS_CONN_RELAY ) {
|
|
|
|
when = comms->r.heartbeat;
|
|
|
|
}
|
2007-11-19 00:43:27 +01:00
|
|
|
#elif defined COMMS_HEARTBEAT
|
2007-12-05 07:33:37 +01:00
|
|
|
if ( comms->doHeartbeat ) {
|
|
|
|
XP_LOGF( "%s: calling util_setTimer", __func__ );
|
|
|
|
when = HB_INTERVAL;
|
|
|
|
} else {
|
|
|
|
XP_LOGF( "%s: doHeartbeat not set", __func__ );
|
|
|
|
}
|
2007-11-19 00:43:27 +01:00
|
|
|
#endif
|
2007-12-05 07:33:37 +01:00
|
|
|
if ( when != 0 ) {
|
|
|
|
util_setTimer( comms->util, TIMER_HEARTBEAT, when,
|
|
|
|
p_comms_timerFired, comms );
|
|
|
|
comms->hbTimerPending = XP_TRUE;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
XP_LOGF( "%s: skipping b/c pending", __func__ );
|
|
|
|
}
|
2007-11-19 00:43:27 +01:00
|
|
|
} /* setHeartbeatTimer */
|
2005-07-23 17:28:15 +02:00
|
|
|
#endif
|
2005-06-23 06:16:53 +02:00
|
|
|
|
2003-11-01 06:35:29 +01:00
|
|
|
#ifdef DEBUG
|
|
|
|
void
|
|
|
|
comms_getStats( CommsCtxt* comms, XWStreamCtxt* stream )
|
|
|
|
{
|
|
|
|
XP_UCHAR buf[100];
|
|
|
|
AddressRecord* rec;
|
2006-09-27 03:54:53 +02:00
|
|
|
MsgQueueElem* elem;
|
2007-11-19 00:43:27 +01:00
|
|
|
XP_U32 now;
|
2003-11-01 06:35:29 +01:00
|
|
|
|
2006-09-27 03:54:53 +02:00
|
|
|
XP_SNPRINTF( (XP_UCHAR*)buf, sizeof(buf),
|
|
|
|
(XP_UCHAR*)"msg queue len: %d\n", comms->queueLen );
|
|
|
|
stream_putString( stream, buf );
|
|
|
|
|
|
|
|
for ( elem = comms->msgQueueHead; !!elem; elem = elem->next ) {
|
|
|
|
XP_SNPRINTF( buf, sizeof(buf),
|
|
|
|
" - channelNo=%d; msgID=" XP_LD "; len=%d\n",
|
|
|
|
elem->channelNo, elem->msgID, elem->len );
|
|
|
|
stream_putString( stream, buf );
|
|
|
|
}
|
2003-11-01 06:35:29 +01:00
|
|
|
|
|
|
|
XP_SNPRINTF( (XP_UCHAR*)buf, sizeof(buf),
|
|
|
|
(XP_UCHAR*)"channel-less bytes sent: %d\n",
|
|
|
|
comms->nUniqueBytes );
|
2006-09-27 03:54:53 +02:00
|
|
|
stream_putString( stream, buf );
|
2003-11-01 06:35:29 +01:00
|
|
|
|
2007-11-19 00:43:27 +01:00
|
|
|
now = util_getCurSeconds( comms->util );
|
2003-11-01 06:35:29 +01:00
|
|
|
for ( rec = comms->recs; !!rec; rec = rec->next ) {
|
|
|
|
XP_SNPRINTF( (XP_UCHAR*)buf, sizeof(buf),
|
|
|
|
(XP_UCHAR*)" Stats for channel: %d\n",
|
|
|
|
rec->channelNo );
|
2006-09-27 03:54:53 +02:00
|
|
|
stream_putString( stream, buf );
|
2003-11-01 06:35:29 +01:00
|
|
|
|
|
|
|
XP_SNPRINTF( (XP_UCHAR*)buf, sizeof(buf),
|
2005-03-20 20:41:30 +01:00
|
|
|
(XP_UCHAR*)"Last msg sent: " XP_LD "\n",
|
2003-11-01 06:35:29 +01:00
|
|
|
rec->nextMsgID );
|
2006-09-27 03:54:53 +02:00
|
|
|
stream_putString( stream, buf );
|
2003-11-01 06:35:29 +01:00
|
|
|
|
|
|
|
XP_SNPRINTF( (XP_UCHAR*)buf, sizeof(buf),
|
|
|
|
(XP_UCHAR*)"Unique bytes sent: %d\n",
|
|
|
|
rec->nUniqueBytes );
|
2006-09-27 03:54:53 +02:00
|
|
|
stream_putString( stream, buf );
|
2003-11-01 06:35:29 +01:00
|
|
|
|
|
|
|
XP_SNPRINTF( (XP_UCHAR*)buf, sizeof(buf),
|
|
|
|
(XP_UCHAR*)"Last message acknowledged: %d\n",
|
|
|
|
rec->lastACK);
|
2006-09-27 03:54:53 +02:00
|
|
|
stream_putString( stream, buf );
|
2007-11-19 00:43:27 +01:00
|
|
|
|
2003-11-01 06:35:29 +01:00
|
|
|
}
|
|
|
|
} /* comms_getStats */
|
|
|
|
#endif
|
|
|
|
|
|
|
|
static AddressRecord*
|
|
|
|
rememberChannelAddress( CommsCtxt* comms, XP_PlayerAddr channelNo,
|
2006-09-14 03:25:40 +02:00
|
|
|
XWHostID hostID, const CommsAddrRec* addr )
|
2003-11-01 06:35:29 +01:00
|
|
|
{
|
|
|
|
AddressRecord* recs = NULL;
|
2008-01-07 02:10:43 +01:00
|
|
|
recs = getRecordFor( comms, NULL, channelNo );
|
2005-03-06 18:56:34 +01:00
|
|
|
if ( !recs ) {
|
|
|
|
/* not found; add a new entry */
|
|
|
|
recs = (AddressRecord*)XP_MALLOC( comms->mpool, sizeof(*recs) );
|
2007-12-05 07:33:37 +01:00
|
|
|
XP_MEMSET( recs, 0, sizeof(*recs) );
|
2005-03-06 18:56:34 +01:00
|
|
|
|
|
|
|
recs->channelNo = channelNo;
|
2006-10-02 16:26:56 +02:00
|
|
|
recs->r.hostID = hostID;
|
2005-03-06 18:56:34 +01:00
|
|
|
recs->next = comms->recs;
|
|
|
|
comms->recs = recs;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* overwrite existing address with new one. I assume that's the right
|
|
|
|
move. */
|
|
|
|
if ( !!recs ) {
|
|
|
|
if ( !!addr ) {
|
|
|
|
XP_MEMCPY( &recs->addr, addr, sizeof(recs->addr) );
|
2006-10-02 16:26:56 +02:00
|
|
|
XP_ASSERT( recs->r.hostID == hostID );
|
2005-03-06 18:56:34 +01:00
|
|
|
} else {
|
|
|
|
XP_MEMSET( &recs->addr, 0, sizeof(recs->addr) );
|
2006-08-26 23:12:10 +02:00
|
|
|
recs->addr.conType = comms->addr.conType;
|
2003-11-01 06:35:29 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
return recs;
|
|
|
|
} /* rememberChannelAddress */
|
|
|
|
|
2007-12-05 07:33:37 +01:00
|
|
|
static void
|
|
|
|
updateChannelAddress( AddressRecord* rec, const CommsAddrRec* addr )
|
|
|
|
{
|
|
|
|
XP_ASSERT( !!rec );
|
|
|
|
XP_MEMCPY( &rec->addr, addr, sizeof(rec->addr) );
|
|
|
|
} /* updateChannelAddress */
|
|
|
|
|
2003-11-01 06:35:29 +01:00
|
|
|
static XP_Bool
|
|
|
|
channelToAddress( CommsCtxt* comms, XP_PlayerAddr channelNo,
|
2007-02-08 03:53:10 +01:00
|
|
|
const CommsAddrRec** addr )
|
2003-11-01 06:35:29 +01:00
|
|
|
{
|
2008-01-07 02:10:43 +01:00
|
|
|
AddressRecord* recs = getRecordFor( comms, NULL, channelNo );
|
2007-02-08 03:53:10 +01:00
|
|
|
XP_Bool found = !!recs;
|
|
|
|
*addr = found? &recs->addr : NULL;
|
|
|
|
return found;
|
2003-11-01 06:35:29 +01:00
|
|
|
} /* channelToAddress */
|
|
|
|
|
|
|
|
static XP_U16
|
2006-10-15 15:53:17 +02:00
|
|
|
countAddrRecs( const CommsCtxt* comms )
|
2003-11-01 06:35:29 +01:00
|
|
|
{
|
|
|
|
short count = 0;
|
|
|
|
AddressRecord* recs;
|
|
|
|
for ( recs = comms->recs; !!recs; recs = recs->next ) {
|
2005-03-19 23:01:38 +01:00
|
|
|
++count;
|
2003-11-01 06:35:29 +01:00
|
|
|
}
|
|
|
|
return count;
|
|
|
|
} /* countAddrRecs */
|
|
|
|
|
2006-10-10 03:34:37 +02:00
|
|
|
#ifdef XWFEATURE_RELAY
|
2006-06-16 03:46:47 +02:00
|
|
|
static XWHostID
|
|
|
|
getDestID( CommsCtxt* comms, XP_PlayerAddr channelNo )
|
|
|
|
{
|
|
|
|
XWHostID id = HOST_ID_NONE;
|
|
|
|
if ( channelNo == CHANNEL_NONE ) {
|
|
|
|
id = HOST_ID_SERVER;
|
|
|
|
} else {
|
|
|
|
AddressRecord* recs;
|
|
|
|
for ( recs = comms->recs; !!recs; recs = recs->next ) {
|
|
|
|
if ( recs->channelNo == channelNo ) {
|
2006-10-02 16:26:56 +02:00
|
|
|
id = recs->r.hostID;
|
2006-06-16 03:46:47 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
XP_LOGF( "getDestID(%d) => %x", channelNo, id );
|
|
|
|
return id;
|
|
|
|
} /* getDestID */
|
|
|
|
|
2005-03-19 23:01:38 +01:00
|
|
|
static XP_Bool
|
2005-06-23 06:16:53 +02:00
|
|
|
send_via_relay( CommsCtxt* comms, XWRELAY_Cmd cmd, XWHostID destID,
|
|
|
|
void* data, int dlen )
|
2005-03-19 23:01:38 +01:00
|
|
|
{
|
2005-06-23 06:16:53 +02:00
|
|
|
XP_Bool success = XP_FALSE;
|
2005-04-03 04:49:04 +02:00
|
|
|
XP_U16 len = 0;
|
2005-03-19 23:01:38 +01:00
|
|
|
CommsAddrRec addr;
|
|
|
|
XWStreamCtxt* tmpStream;
|
|
|
|
XP_U8* buf;
|
|
|
|
|
|
|
|
comms_getAddr( comms, &addr );
|
|
|
|
tmpStream = mem_stream_make( MPPARM(comms->mpool)
|
|
|
|
util_getVTManager(comms->util),
|
|
|
|
NULL, 0,
|
|
|
|
(MemStreamCloseCallback)NULL );
|
|
|
|
if ( tmpStream != NULL ) {
|
|
|
|
stream_open( tmpStream );
|
|
|
|
stream_putU8( tmpStream, cmd );
|
|
|
|
|
2005-07-06 02:58:20 +02:00
|
|
|
switch ( cmd ) {
|
|
|
|
case XWRELAY_MSG_TORELAY:
|
2006-10-02 16:26:56 +02:00
|
|
|
stream_putU16( tmpStream, comms->r.cookieID );
|
|
|
|
stream_putU8( tmpStream, comms->r.myHostID );
|
2005-10-01 18:01:39 +02:00
|
|
|
stream_putU8( tmpStream, destID );
|
2005-03-19 23:01:38 +01:00
|
|
|
if ( data != NULL && dlen > 0 ) {
|
|
|
|
stream_putBytes( tmpStream, data, dlen );
|
|
|
|
}
|
2005-07-06 02:58:20 +02:00
|
|
|
break;
|
2005-09-05 17:33:51 +02:00
|
|
|
case XWRELAY_GAME_CONNECT:
|
2005-07-06 03:36:52 +02:00
|
|
|
stream_putU8( tmpStream, XWRELAY_PROTO_VERSION );
|
2005-07-06 02:58:20 +02:00
|
|
|
stringToStream( tmpStream, addr.u.ip_relay.cookie );
|
2006-10-02 16:26:56 +02:00
|
|
|
stream_putU8( tmpStream, comms->r.myHostID );
|
|
|
|
stream_putU8( tmpStream, comms->r.nPlayersHere );
|
|
|
|
stream_putU8( tmpStream, comms->r.nPlayersTotal );
|
2005-03-19 23:01:38 +01:00
|
|
|
|
2006-10-02 16:26:56 +02:00
|
|
|
comms->r.relayState = COMMS_RELAYSTATE_CONNECT_PENDING;
|
2005-07-06 02:58:20 +02:00
|
|
|
break;
|
|
|
|
|
2005-09-05 17:33:51 +02:00
|
|
|
case XWRELAY_GAME_RECONNECT:
|
2005-09-02 08:26:33 +02:00
|
|
|
stream_putU8( tmpStream, XWRELAY_PROTO_VERSION );
|
2006-10-02 16:26:56 +02:00
|
|
|
stream_putU8( tmpStream, comms->r.myHostID );
|
|
|
|
stream_putU8( tmpStream, comms->r.nPlayersHere );
|
|
|
|
stream_putU8( tmpStream, comms->r.nPlayersTotal );
|
|
|
|
stringToStream( tmpStream, comms->r.connName );
|
2005-09-02 08:26:33 +02:00
|
|
|
|
2006-10-02 16:26:56 +02:00
|
|
|
comms->r.relayState = COMMS_RELAYSTATE_CONNECT_PENDING;
|
2005-09-02 08:26:33 +02:00
|
|
|
break;
|
|
|
|
|
2005-09-05 17:33:51 +02:00
|
|
|
case XWRELAY_GAME_DISCONNECT:
|
2006-10-02 16:26:56 +02:00
|
|
|
stream_putU16( tmpStream, comms->r.cookieID );
|
|
|
|
stream_putU8( tmpStream, comms->r.myHostID );
|
2005-09-05 17:33:51 +02:00
|
|
|
break;
|
|
|
|
|
2007-11-19 00:43:27 +01:00
|
|
|
#ifdef RELAY_HEARTBEAT
|
2005-07-06 02:58:20 +02:00
|
|
|
case XWRELAY_HEARTBEAT:
|
2005-06-23 06:16:53 +02:00
|
|
|
/* Add these for grins. Server can assert they match the IP
|
|
|
|
address it expects 'em on. */
|
2006-10-02 16:26:56 +02:00
|
|
|
stream_putU16( tmpStream, comms->r.cookieID );
|
|
|
|
stream_putU8( tmpStream, comms->r.myHostID );
|
2005-07-06 02:58:20 +02:00
|
|
|
break;
|
2007-11-19 00:43:27 +01:00
|
|
|
#endif
|
2005-09-02 08:26:33 +02:00
|
|
|
default:
|
|
|
|
XP_ASSERT(0);
|
2005-03-19 23:01:38 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
len = stream_getSize( tmpStream );
|
|
|
|
buf = XP_MALLOC( comms->mpool, len );
|
|
|
|
if ( buf != NULL ) {
|
|
|
|
stream_getBytes( tmpStream, buf, len );
|
|
|
|
}
|
|
|
|
stream_destroy( tmpStream );
|
|
|
|
if ( buf != NULL ) {
|
2005-06-23 06:16:53 +02:00
|
|
|
XP_U16 result;
|
2005-03-19 23:01:38 +01:00
|
|
|
XP_LOGF( "passing %d bytes to sendproc", len );
|
|
|
|
result = (*comms->sendproc)( buf, len, &addr, comms->sendClosure );
|
2005-06-23 06:16:53 +02:00
|
|
|
success = result == len;
|
|
|
|
if ( success ) {
|
|
|
|
setHeartbeatTimer( comms );
|
|
|
|
}
|
2005-03-19 23:01:38 +01:00
|
|
|
}
|
|
|
|
XP_FREE( comms->mpool, buf );
|
|
|
|
}
|
2005-06-23 06:16:53 +02:00
|
|
|
return success;
|
|
|
|
} /* send_via_relay */
|
2005-03-19 23:01:38 +01:00
|
|
|
|
|
|
|
/* Send a CONNECT message to the relay. This opens up a connection to the
|
|
|
|
* relay, and tells it our hostID and cookie so that it can associatate it
|
|
|
|
* with a socket. In the CONNECT_RESP we should get back what?
|
|
|
|
*/
|
|
|
|
static void
|
2005-06-23 06:16:53 +02:00
|
|
|
relayConnect( CommsCtxt* comms )
|
2005-03-19 23:01:38 +01:00
|
|
|
{
|
2007-02-06 06:49:45 +01:00
|
|
|
LOG_FUNC();
|
2006-10-02 16:26:56 +02:00
|
|
|
if ( comms->addr.conType == COMMS_CONN_RELAY && !comms->r.connecting ) {
|
|
|
|
comms->r.connecting = XP_TRUE;
|
2005-09-02 08:26:33 +02:00
|
|
|
send_via_relay( comms,
|
2006-10-02 16:26:56 +02:00
|
|
|
comms->r.connName[0] == '\0' ?
|
2005-09-05 17:33:51 +02:00
|
|
|
XWRELAY_GAME_CONNECT:XWRELAY_GAME_RECONNECT,
|
2006-10-02 16:26:56 +02:00
|
|
|
comms->r.myHostID, NULL, 0 );
|
|
|
|
comms->r.connecting = XP_FALSE;
|
2005-04-03 04:49:04 +02:00
|
|
|
}
|
2005-06-23 06:16:53 +02:00
|
|
|
} /* relayConnect */
|
2006-10-10 03:34:37 +02:00
|
|
|
#endif
|
2005-09-05 17:33:51 +02:00
|
|
|
|
2007-11-26 03:58:25 +01:00
|
|
|
#if defined XWFEATURE_BLUETOOTH || defined XWFEATURE_IP_DIRECT
|
2007-02-08 03:53:10 +01:00
|
|
|
static XP_S16
|
2007-11-26 03:58:25 +01:00
|
|
|
send_via_bt_or_ip( CommsCtxt* comms, BTIPMsgType typ, XP_PlayerAddr channelNo,
|
|
|
|
void* data, int dlen )
|
2007-02-08 03:53:10 +01:00
|
|
|
{
|
2007-12-05 07:33:37 +01:00
|
|
|
XP_S16 nSent;
|
2007-02-08 03:53:10 +01:00
|
|
|
XP_U8* buf;
|
2007-12-05 07:33:37 +01:00
|
|
|
LOG_FUNC();
|
|
|
|
nSent = -1;
|
2007-02-08 03:53:10 +01:00
|
|
|
buf = XP_MALLOC( comms->mpool, dlen + 1 );
|
|
|
|
if ( !!buf ) {
|
|
|
|
const CommsAddrRec* addr;
|
|
|
|
(void)channelToAddress( comms, channelNo, &addr );
|
|
|
|
|
|
|
|
buf[0] = typ;
|
|
|
|
if ( dlen > 0 ) {
|
|
|
|
XP_MEMCPY( &buf[1], data, dlen );
|
|
|
|
}
|
|
|
|
|
|
|
|
nSent = (*comms->sendproc)( buf, dlen+1, addr, comms->sendClosure );
|
|
|
|
XP_FREE( comms->mpool, buf );
|
2007-11-19 00:43:27 +01:00
|
|
|
|
|
|
|
setHeartbeatTimer( comms );
|
2007-02-08 03:53:10 +01:00
|
|
|
}
|
2007-12-05 07:33:37 +01:00
|
|
|
LOG_RETURNF( "%d", nSent );
|
2007-02-08 03:53:10 +01:00
|
|
|
return nSent;
|
2007-11-26 03:58:25 +01:00
|
|
|
} /* send_via_bt_or_ip */
|
2007-02-08 03:53:10 +01:00
|
|
|
|
2006-08-26 23:12:10 +02:00
|
|
|
#endif
|
|
|
|
|
2006-10-10 03:34:37 +02:00
|
|
|
#ifdef XWFEATURE_RELAY
|
2005-09-05 17:33:51 +02:00
|
|
|
static void
|
|
|
|
relayDisconnect( CommsCtxt* comms )
|
|
|
|
{
|
|
|
|
XP_LOGF( "relayDisconnect called" );
|
2006-08-23 06:44:55 +02:00
|
|
|
if ( comms->addr.conType == COMMS_CONN_RELAY ) {
|
2006-10-02 16:26:56 +02:00
|
|
|
if ( comms->r.relayState != COMMS_RELAYSTATE_UNCONNECTED ) {
|
|
|
|
comms->r.relayState = COMMS_RELAYSTATE_UNCONNECTED;
|
2006-08-26 23:12:10 +02:00
|
|
|
send_via_relay( comms, XWRELAY_GAME_DISCONNECT, HOST_ID_NONE,
|
|
|
|
NULL, 0 );
|
2006-08-23 06:44:55 +02:00
|
|
|
}
|
2005-09-05 17:33:51 +02:00
|
|
|
}
|
|
|
|
} /* relayDisconnect */
|
2005-07-23 17:28:15 +02:00
|
|
|
#endif
|
2005-03-19 23:01:38 +01:00
|
|
|
|
2005-01-31 04:31:50 +01:00
|
|
|
EXTERN_C_END
|
|
|
|
|
2003-11-01 06:35:29 +01:00
|
|
|
#endif /* #ifndef XWFEATURE_STANDALONE_ONLY */
|