2011-09-22 17:31:57 -07:00
|
|
|
/* -*- compile-command: "cd ../linux && make MEMDEBUG=TRUE -j3"; -*- */
|
2003-11-01 05:35:29 +00:00
|
|
|
/*
|
2023-03-27 14:29:49 -07:00
|
|
|
* Copyright 2001 - 2023 by Eric House (xwords@eehouse.org). All rights
|
2009-01-03 23:54:25 +00:00
|
|
|
* reserved.
|
2003-11-01 05:35:29 +00: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"
|
2018-07-05 07:58:50 -07:00
|
|
|
#include "dutil.h"
|
2004-10-08 23:54:57 +00:00
|
|
|
#include "game.h"
|
2003-11-01 05:35:29 +00:00
|
|
|
#include "xwstream.h"
|
|
|
|
#include "memstream.h"
|
2005-03-19 22:01:38 +00:00
|
|
|
#include "xwrelay.h"
|
2005-03-06 17:56:34 +00:00
|
|
|
#include "strutils.h"
|
2017-01-22 14:15:29 -08:00
|
|
|
#include "dbgutil.h"
|
2020-09-21 11:31:49 -07:00
|
|
|
#include "knownplyr.h"
|
2022-09-04 09:59:47 -07:00
|
|
|
#include "device.h"
|
2022-09-06 12:16:29 -07:00
|
|
|
#include "nli.h"
|
2003-11-01 05:35:29 +00:00
|
|
|
|
2005-06-23 04:16:53 +00:00
|
|
|
#define HEARTBEAT_NONE 0
|
2003-11-01 05:35:29 +00:00
|
|
|
|
2015-02-18 19:19:36 -08:00
|
|
|
#define HAS_VERSION_FLAG 0xBEEF
|
2021-12-29 07:35:43 -08:00
|
|
|
#define COMMS_VERSION 2
|
2021-12-21 16:23:24 -08:00
|
|
|
|
2021-12-29 07:35:43 -08:00
|
|
|
/* Flags: 7 bits of header len, 6 bits of flags (3 used so far), and 3 bits of
|
2021-12-21 16:23:24 -08:00
|
|
|
comms version */
|
2021-12-29 07:35:43 -08:00
|
|
|
#define VERSION_MASK 0x0007
|
|
|
|
#define NO_CONNID_BIT 0x0008
|
2015-02-23 18:28:31 -08:00
|
|
|
#define IS_SERVER_BIT 0x0010
|
2021-12-28 18:49:56 -08:00
|
|
|
#define NO_MSGID_BIT 0x0020
|
2021-12-21 16:23:24 -08:00
|
|
|
#define HEADER_LEN_OFFSET 9
|
2015-02-18 19:19:36 -08:00
|
|
|
|
2022-09-06 12:16:29 -07:00
|
|
|
/* Low two bits treated as channel, third as short-term flag indicating
|
|
|
|
* sender's role; rest can be random to aid detection of duplicate packets. */
|
|
|
|
#define CHANNEL_MASK 0x0003
|
|
|
|
|
2003-11-01 05:35:29 +00:00
|
|
|
#ifndef XWFEATURE_STANDALONE_ONLY
|
|
|
|
|
2012-01-26 05:43:54 -08:00
|
|
|
#ifndef INITIAL_CLIENT_VERS
|
2012-05-07 07:02:43 -07:00
|
|
|
# define INITIAL_CLIENT_VERS 2
|
2012-01-26 05:43:54 -08:00
|
|
|
#endif
|
|
|
|
|
2023-01-06 07:52:41 -08:00
|
|
|
#ifdef DEBUG
|
|
|
|
typedef struct StackData {
|
|
|
|
const char* func;
|
|
|
|
pthread_t prevThread;
|
|
|
|
struct StackData* prev;
|
|
|
|
int count;
|
|
|
|
} StackData;
|
|
|
|
|
|
|
|
static void
|
|
|
|
printStack( const StackData* data )
|
|
|
|
{
|
|
|
|
for ( ; !!data; data = data->prev ) {
|
|
|
|
XP_LOGFF( " proc[%d]: %s", data->count, data->func );
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
typedef struct _SD {
|
|
|
|
StackData* head;
|
|
|
|
pthread_t thread;
|
|
|
|
} _SD;
|
|
|
|
|
|
|
|
# define OBJ_DECL _SD _sd
|
|
|
|
|
|
|
|
# define THREAD_CHECK_START(OBJ) { \
|
|
|
|
_SD* _sdp = &(OBJ)->_sd; \
|
|
|
|
StackData _data = { .func = __func__, \
|
|
|
|
.prev = _sdp->head, \
|
|
|
|
.prevThread = _sdp->thread, \
|
|
|
|
}; \
|
|
|
|
_data.count = NULL == _data.prev ? 0 : _data.prev->count + 1; \
|
|
|
|
\
|
|
|
|
(OBJ)->_sd.head = &_data; \
|
|
|
|
pthread_t thread = pthread_self(); \
|
|
|
|
if ( 0 == (OBJ)->_sd.thread ) { \
|
|
|
|
(OBJ)->_sd.thread = thread; \
|
|
|
|
} else if ( thread != (OBJ)->_sd.thread ) { \
|
|
|
|
printStack( &_data ); \
|
|
|
|
XP_ASSERT(0); \
|
|
|
|
} \
|
|
|
|
|
|
|
|
# define THREAD_CHECK_END() \
|
|
|
|
_sdp->head = _data.prev; \
|
|
|
|
_sdp->thread = _data.prevThread; \
|
|
|
|
} \
|
|
|
|
|
|
|
|
#else
|
|
|
|
# define OBJ_DECL
|
|
|
|
# define THREAD_CHECK_START(OBJ)
|
|
|
|
# define THREAD_CHECK_END()
|
|
|
|
#endif
|
|
|
|
|
2007-11-18 23:43:27 +00:00
|
|
|
#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 06:33:37 +00:00
|
|
|
# ifndef HB_INTERVAL
|
|
|
|
# define HB_INTERVAL 5
|
|
|
|
# endif
|
2007-11-18 23:43:27 +00:00
|
|
|
#endif
|
|
|
|
|
2005-01-31 03:31:50 +00:00
|
|
|
EXTERN_C_START
|
|
|
|
|
2003-11-01 05:35:29 +00:00
|
|
|
typedef struct MsgQueueElem {
|
2023-03-18 15:15:34 -07:00
|
|
|
SendMsgsPacket smp;
|
2007-12-05 06:33:37 +00:00
|
|
|
XP_PlayerAddr channelNo;
|
2012-11-13 21:18:52 -08:00
|
|
|
#ifdef DEBUG
|
2007-11-26 02:58:25 +00:00
|
|
|
XP_U16 sendCount; /* how many times sent? */
|
2012-11-13 21:18:52 -08:00
|
|
|
#endif
|
2005-03-19 22:01:38 +00:00
|
|
|
MsgID msgID; /* saved for ease of deletion */
|
2010-10-18 22:08:04 -07:00
|
|
|
#ifdef COMMS_CHECKSUM
|
2013-07-10 08:05:47 -07:00
|
|
|
XP_UCHAR* checksum;
|
2010-10-18 20:17:44 -07:00
|
|
|
#endif
|
2003-11-01 05:35:29 +00:00
|
|
|
} MsgQueueElem;
|
|
|
|
|
|
|
|
typedef struct AddressRecord {
|
|
|
|
struct AddressRecord* next;
|
2023-03-16 07:41:53 -07:00
|
|
|
MsgQueueElem* _msgQueueHead;
|
|
|
|
|
2003-11-01 05:35:29 +00:00
|
|
|
CommsAddrRec addr;
|
2008-05-31 03:26:16 +00:00
|
|
|
MsgID nextMsgID; /* on a per-channel basis */
|
2012-04-13 20:21:59 -07:00
|
|
|
MsgID lastMsgAckd; /* on a per-channel basis */
|
2012-09-10 07:31:45 -07:00
|
|
|
|
|
|
|
/* lastMsgRcd is the numerically highest MsgID we've seen. Because once
|
|
|
|
* it's sent in message as an ACK the other side will delete messages
|
|
|
|
* based on it, we don't send a number higher than has actually been
|
|
|
|
* written out successfully. lastMsgSaved is that number.
|
|
|
|
*/
|
|
|
|
MsgID lastMsgRcd;
|
|
|
|
MsgID lastMsgSaved;
|
2007-11-18 23:43:27 +00:00
|
|
|
/* only used if COMMS_HEARTBEAT set except for serialization (to_stream) */
|
2003-11-01 05:35:29 +00:00
|
|
|
XP_PlayerAddr channelNo;
|
2021-12-29 07:35:43 -08:00
|
|
|
XP_U16 flags; /* storing only COMMS_VERSION */
|
2006-10-02 14:26:56 +00:00
|
|
|
struct {
|
|
|
|
XWHostID hostID; /* used for relay case */
|
2014-02-07 21:12:45 -08:00
|
|
|
} rr;
|
2007-12-05 06:33:37 +00:00
|
|
|
#ifdef COMMS_HEARTBEAT
|
|
|
|
XP_Bool initialSeen;
|
|
|
|
#endif
|
2003-11-01 05:35:29 +00:00
|
|
|
} AddressRecord;
|
|
|
|
|
2004-10-08 23:54:57 +00:00
|
|
|
#define ADDRESSRECORD_SIZE_68K 20
|
|
|
|
|
2003-11-01 05:35:29 +00:00
|
|
|
struct CommsCtxt {
|
|
|
|
XW_UtilCtxt* util;
|
2018-07-05 07:58:50 -07:00
|
|
|
XW_DUtilCtxt* dutil;
|
2003-11-01 05:35:29 +00:00
|
|
|
|
2009-08-21 12:00:09 +00:00
|
|
|
XP_U32 connID; /* set from gameID: 0 means ignore; otherwise
|
|
|
|
must match. Set by server. */
|
2023-02-05 08:59:37 -08:00
|
|
|
XP_U16 streamVersion; /* negotiated by server */
|
2007-12-05 06:33:37 +00:00
|
|
|
XP_PlayerAddr nextChannelNo;
|
2005-03-19 22:01:38 +00:00
|
|
|
|
2003-11-01 05:35:29 +00:00
|
|
|
AddressRecord* recs; /* return addresses */
|
|
|
|
|
2009-09-12 21:39:13 +00:00
|
|
|
TransportProcs procs;
|
2020-08-25 14:47:56 -07:00
|
|
|
|
|
|
|
RoleChangeProc rcProc;
|
|
|
|
void* rcClosure;
|
|
|
|
|
2011-08-16 19:36:23 -07:00
|
|
|
XP_U32 xportFlags;
|
2007-11-18 23:43:27 +00:00
|
|
|
#ifdef COMMS_HEARTBEAT
|
2012-01-25 18:27:37 -08:00
|
|
|
XP_U32 lastMsgRcd;
|
2007-11-18 23:43:27 +00:00
|
|
|
#endif
|
2003-11-01 05:35:29 +00:00
|
|
|
void* sendClosure;
|
|
|
|
XP_U16 queueLen;
|
2009-09-26 14:37:49 +00:00
|
|
|
XP_U16 channelSeed; /* tries to be unique per device to aid
|
|
|
|
dupe elimination at start */
|
2021-12-21 16:23:24 -08:00
|
|
|
XP_U32 nextResend; /* timestamp */
|
2012-11-13 21:25:03 -08:00
|
|
|
XP_U16 resendBackoff;
|
|
|
|
|
2007-11-18 23:43:27 +00:00
|
|
|
#ifdef COMMS_HEARTBEAT
|
|
|
|
XP_Bool doHeartbeat;
|
2007-12-05 06:33:37 +00:00
|
|
|
XP_U32 lastMsgRcvdTime;
|
2007-11-18 23:43:27 +00:00
|
|
|
#endif
|
2009-02-01 15:50:58 +00:00
|
|
|
#if defined XWFEATURE_RELAY || defined COMMS_HEARTBEAT
|
|
|
|
XP_Bool hbTimerPending;
|
|
|
|
XP_Bool reconTimerPending;
|
|
|
|
#endif
|
2012-09-10 07:31:45 -07:00
|
|
|
XP_U16 lastSaveToken;
|
2014-12-21 17:40:00 -08:00
|
|
|
XP_U16 forceChannel;
|
2007-11-18 23:43:27 +00:00
|
|
|
|
2006-10-10 01:34:37 +00: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.*/
|
2022-09-03 19:14:40 -07:00
|
|
|
CommsAddrRec selfAddr;
|
2003-11-01 05:35:29 +00:00
|
|
|
|
2005-03-19 22:01:38 +00:00
|
|
|
/* Stuff for relays */
|
2006-10-02 14:26:56 +00:00
|
|
|
struct {
|
2009-02-28 19:27:45 +00:00
|
|
|
XWHostID myHostID; /* 0 if unset, 1 if acting as server.
|
|
|
|
Client's 0 replaced by id assigned by
|
|
|
|
relay. Relay calls this "srcID". */
|
2006-10-07 03:37:40 +00:00
|
|
|
CommsRelayState relayState; /* not saved: starts at UNCONNECTED */
|
2006-10-02 14:26:56 +00: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;
|
2014-02-07 21:12:45 -08:00
|
|
|
} rr;
|
2006-04-07 03:08:23 +00:00
|
|
|
|
2020-09-21 11:31:49 -07:00
|
|
|
XP_U8 flags;
|
|
|
|
|
2003-11-01 05:35:29 +00:00
|
|
|
XP_Bool isServer;
|
2014-11-06 06:35:28 -08:00
|
|
|
XP_Bool disableds[COMMS_CONN_NTYPES][2];
|
2017-05-26 07:30:42 -07:00
|
|
|
#ifdef DEBUG
|
2015-02-08 07:51:08 -08:00
|
|
|
XP_Bool processingMsg;
|
2015-02-09 19:15:43 -08:00
|
|
|
const XP_UCHAR* tag;
|
2014-11-06 06:35:28 -08:00
|
|
|
#endif
|
2023-01-06 07:52:41 -08:00
|
|
|
OBJ_DECL;
|
2003-11-01 05:35:29 +00:00
|
|
|
MPSLOT
|
|
|
|
};
|
|
|
|
|
2020-09-21 11:31:49 -07:00
|
|
|
#define FLAG_HARVEST_DONE 1
|
2023-11-05 11:52:48 -08:00
|
|
|
#define FLAG_QUASHED 2
|
2023-12-18 17:08:51 -08:00
|
|
|
|
2023-11-05 11:52:48 -08:00
|
|
|
#define QUASHED(COMMS) (0 != ((COMMS)->flags & FLAG_QUASHED))
|
2020-09-21 11:31:49 -07:00
|
|
|
|
2014-10-15 07:26:18 -07:00
|
|
|
#if defined XWFEATURE_IP_DIRECT || defined XWFEATURE_DIRECTIP
|
2007-02-08 02:53:10 +00:00
|
|
|
typedef enum {
|
2007-11-26 02:58:25 +00:00
|
|
|
BTIPMSG_NONE = 0
|
|
|
|
,BTIPMSG_DATA
|
|
|
|
,BTIPMSG_RESET
|
2007-12-05 06:33:37 +00:00
|
|
|
,BTIPMSG_HB
|
2007-11-26 02:58:25 +00:00
|
|
|
} BTIPMsgType;
|
2007-02-08 02:53:10 +00:00
|
|
|
#endif
|
|
|
|
|
2021-12-15 10:03:09 -08:00
|
|
|
#define TAGFMT(...) "<%s> (" #__VA_ARGS__ "): "
|
|
|
|
#define TAGPRMS comms->tag
|
2015-02-09 19:15:43 -08:00
|
|
|
|
2003-11-01 05:35:29 +00:00
|
|
|
/****************************************************************************
|
|
|
|
* prototypes
|
|
|
|
****************************************************************************/
|
2023-02-15 08:31:51 -08:00
|
|
|
static AddressRecord* rememberChannelAddress( CommsCtxt* comms,
|
|
|
|
XP_PlayerAddr channelNo,
|
|
|
|
XWHostID id,
|
|
|
|
const CommsAddrRec* addr,
|
|
|
|
XP_U16 flags );
|
2020-09-10 15:11:22 -07:00
|
|
|
static void augmentChannelAddr( CommsCtxt* comms, AddressRecord* rec,
|
|
|
|
const CommsAddrRec* addr, XWHostID hostID );
|
2020-09-24 16:37:50 -07:00
|
|
|
static XP_Bool augmentAddrIntrnl( CommsCtxt* comms, CommsAddrRec* dest,
|
2020-09-25 10:17:47 -07:00
|
|
|
const CommsAddrRec* src, XP_Bool isNewer );
|
2023-02-15 08:31:51 -08:00
|
|
|
static XP_Bool channelToAddress( const CommsCtxt* comms, XP_PlayerAddr channelNo,
|
|
|
|
const CommsAddrRec** addr );
|
|
|
|
static AddressRecord* getRecordFor( const CommsCtxt* comms, XP_PlayerAddr channelNo );
|
2023-01-06 07:52:41 -08:00
|
|
|
static XP_S16 sendMsg( const CommsCtxt* comms, XWEnv xwe, MsgQueueElem* elem,
|
2020-04-25 17:47:07 -07:00
|
|
|
CommsConnType filter );
|
2023-02-01 19:08:04 -08:00
|
|
|
static MsgQueueElem* addToQueue( CommsCtxt* comms, XWEnv xwe,
|
|
|
|
MsgQueueElem* newElem, XP_Bool notify );
|
2013-07-12 07:34:12 -07:00
|
|
|
static XP_Bool elems_same( const MsgQueueElem* e1, const MsgQueueElem* e2 ) ;
|
2022-09-28 07:24:45 -07:00
|
|
|
static void freeElem( MPFORMAL MsgQueueElem* elem );
|
2022-09-08 09:57:11 -07:00
|
|
|
static void removeFromQueue( CommsCtxt* comms, XWEnv xwe, XP_PlayerAddr channelNo,
|
|
|
|
MsgID msgID );
|
2006-10-15 13:53:17 +00:00
|
|
|
static XP_U16 countAddrRecs( const CommsCtxt* comms );
|
2022-05-14 07:10:48 -07:00
|
|
|
static void sendConnect( CommsCtxt* comms, XWEnv xwe
|
|
|
|
#ifdef XWFEATURE_RELAY
|
|
|
|
, XP_Bool breakExisting
|
|
|
|
#endif
|
|
|
|
);
|
2023-02-01 19:08:04 -08:00
|
|
|
|
|
|
|
typedef enum {FEA_OK = 0x00, FEA_REMOVE = 0x01, FEA_EXIT = 0x02} ForEachAct;
|
|
|
|
typedef ForEachAct (*EachMsgProc)( MsgQueueElem* elem, void* closure );
|
|
|
|
static void forEachElem( CommsCtxt* comms, EachMsgProc proc, void* closure );
|
|
|
|
|
|
|
|
static MsgQueueElem* makeNewElem( const CommsCtxt* comms, XWEnv xwe, MsgID msgID,
|
|
|
|
XP_PlayerAddr channelNo );
|
2020-04-25 17:47:07 -07:00
|
|
|
static void notifyQueueChanged( const CommsCtxt* comms, XWEnv xwe );
|
2022-09-06 12:16:29 -07:00
|
|
|
static XP_U16 makeFlags( const CommsCtxt* comms, XP_U16 headerLen,
|
|
|
|
MsgID msgID );
|
2021-03-19 18:51:34 -07:00
|
|
|
|
2022-02-12 13:27:24 -08:00
|
|
|
static XP_Bool formatRelayID( const CommsCtxt* comms, XWHostID hostID,
|
|
|
|
XP_UCHAR* buf, XP_U16* lenp );
|
|
|
|
|
2006-10-10 01:34:37 +00:00
|
|
|
#ifdef XWFEATURE_RELAY
|
2022-04-28 07:37:33 -07:00
|
|
|
static void set_reset_timer( CommsCtxt* comms, XWEnv xwe );
|
|
|
|
static XP_Bool sendNoConn( CommsCtxt* comms, XWEnv xwe,
|
|
|
|
const MsgQueueElem* elem, XWHostID destID );
|
2020-04-25 17:47:07 -07:00
|
|
|
static XP_Bool relayConnect( CommsCtxt* comms, XWEnv xwe );
|
|
|
|
static void relayDisconnect( CommsCtxt* comms, XWEnv xwe );
|
|
|
|
static XP_Bool send_via_relay( CommsCtxt* comms, XWEnv xwe, XWRELAY_Cmd cmd,
|
2015-09-30 06:50:04 -07:00
|
|
|
XWHostID destID, void* data, int dlen,
|
|
|
|
const XP_UCHAR* msgNo );
|
2020-04-25 17:47:07 -07:00
|
|
|
static XWHostID getDestID( CommsCtxt* comms, XWEnv xwe, XP_PlayerAddr channelNo );
|
2012-10-30 07:01:47 -07:00
|
|
|
# ifdef XWFEATURE_DEVID
|
2020-04-26 11:28:18 -07:00
|
|
|
static void putDevID( const CommsCtxt* comms, XWEnv xwe, XWStreamCtxt* stream );
|
2022-09-08 09:57:11 -07:00
|
|
|
|
|
|
|
|
2012-10-30 07:01:47 -07:00
|
|
|
# else
|
2020-04-26 11:28:18 -07:00
|
|
|
# define putDevID( comms, xwe, stream )
|
2012-10-30 07:01:47 -07:00
|
|
|
# endif
|
2021-03-19 18:51:34 -07:00
|
|
|
# else
|
|
|
|
# define relayDisconnect( comms, xwe )
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#ifdef DEBUG
|
2022-09-03 18:47:04 -07:00
|
|
|
static void assertAddrOk( const CommsAddrRec* addr );
|
2022-10-07 09:23:09 -07:00
|
|
|
static void listRecs( const CommsCtxt* comms, const char* msg );
|
2023-03-15 12:26:56 -07:00
|
|
|
static void _assertQueueOk( const CommsCtxt* comms, const char* func );
|
|
|
|
# define assertQueueOk(COM) _assertQueueOk((COM), __func__)
|
2023-02-01 19:08:04 -08:00
|
|
|
|
2022-09-03 18:47:04 -07:00
|
|
|
#define ASSERT_ADDR_OK(addr) assertAddrOk( addr )
|
|
|
|
|
2022-04-28 07:37:33 -07:00
|
|
|
# ifdef XWFEATURE_RELAY
|
2009-11-22 18:00:20 +00:00
|
|
|
static const char* relayCmdToStr( XWRELAY_Cmd cmd );
|
2022-04-28 07:37:33 -07:00
|
|
|
# endif
|
2013-07-07 19:40:07 -07:00
|
|
|
static void printQueue( const CommsCtxt* comms );
|
2023-02-16 21:36:46 -08:00
|
|
|
static void logAddrComms( const CommsCtxt* comms, const CommsAddrRec* addr,
|
|
|
|
const char* caller );
|
2021-03-19 18:51:34 -07:00
|
|
|
#else
|
2022-11-12 12:38:00 -08:00
|
|
|
# define ASSERT_ADDR_OK(addr)
|
2023-02-01 19:08:04 -08:00
|
|
|
# define assertQueueOk(comms)
|
2022-11-12 12:38:00 -08:00
|
|
|
# define printQueue( comms )
|
2023-02-05 08:59:37 -08:00
|
|
|
# define logAddr( comms, addr, caller)
|
2022-11-18 11:04:59 -08:00
|
|
|
# define listRecs( comms, caller )
|
2023-04-06 20:02:30 -07:00
|
|
|
# define logAddrComms(comms, addr, caller)
|
2022-11-12 12:38:00 -08:00
|
|
|
#endif /* def DEBUG */
|
|
|
|
|
2012-04-13 20:21:59 -07:00
|
|
|
#if defined RELAY_HEARTBEAT || defined COMMS_HEARTBEAT
|
2005-06-23 04:16:53 +00:00
|
|
|
static void setHeartbeatTimer( CommsCtxt* comms );
|
2009-02-01 15:50:58 +00:00
|
|
|
|
2007-11-26 02:58:25 +00:00
|
|
|
#else
|
|
|
|
# define setHeartbeatTimer( comms )
|
2006-10-10 01:34:37 +00:00
|
|
|
#endif
|
2014-10-15 07:26:18 -07:00
|
|
|
#if defined XWFEATURE_IP_DIRECT || defined XWFEATURE_DIRECTIP
|
2020-04-25 17:47:07 -07:00
|
|
|
static XP_S16 send_via_bt_or_ip( CommsCtxt* comms, XWEnv xwe, BTIPMsgType msgTyp,
|
2007-11-26 02:58:25 +00:00
|
|
|
XP_PlayerAddr channelNo,
|
2014-10-15 07:26:18 -07:00
|
|
|
CommsConnType typ,
|
2015-09-30 06:50:04 -07:00
|
|
|
void* data, int dlen, const XP_UCHAR* msgNo );
|
2006-06-16 01:46:47 +00:00
|
|
|
#endif
|
2003-11-01 05:35:29 +00:00
|
|
|
|
2012-04-13 20:57:35 -07:00
|
|
|
#if defined COMMS_HEARTBEAT || defined XWFEATURE_COMMSACK
|
2023-10-26 12:25:59 -07:00
|
|
|
static void sendEmptyMsg( CommsCtxt* comms, XWEnv xwe, AddressRecord* rec );
|
2012-04-13 20:57:35 -07:00
|
|
|
#endif
|
2022-09-08 09:57:11 -07:00
|
|
|
static inline XP_Bool IS_INVITE(const MsgQueueElem* elem)
|
|
|
|
{
|
2023-03-18 15:15:34 -07:00
|
|
|
return 0 == elem->smp.len;
|
2022-09-08 09:57:11 -07:00
|
|
|
}
|
2012-04-13 20:57:35 -07:00
|
|
|
|
2003-11-01 05:35:29 +00:00
|
|
|
/****************************************************************************
|
|
|
|
* implementation
|
|
|
|
****************************************************************************/
|
2009-08-21 12:00:09 +00:00
|
|
|
|
2014-11-06 18:44:19 -08:00
|
|
|
#ifdef DEBUG
|
|
|
|
# define CNO_FMT(buf, cno) \
|
|
|
|
XP_UCHAR (buf)[64]; \
|
2015-02-08 09:58:34 -08:00
|
|
|
XP_SNPRINTF( (buf), sizeof(buf), "cno: %.4X|%x", \
|
2014-11-06 18:44:19 -08:00
|
|
|
(cno) & ~CHANNEL_MASK, (cno) & CHANNEL_MASK )
|
|
|
|
#else
|
|
|
|
# define CNO_FMT(buf, cno)
|
|
|
|
#endif
|
|
|
|
|
2021-03-19 18:51:34 -07:00
|
|
|
#ifdef XWFEATURE_RELAY
|
|
|
|
|
2009-08-21 12:00:09 +00:00
|
|
|
#ifdef DEBUG
|
2009-09-12 21:39:13 +00:00
|
|
|
const char*
|
2009-08-21 12:00:09 +00:00
|
|
|
CommsRelayState2Str( CommsRelayState state )
|
|
|
|
{
|
|
|
|
#define CASE_STR(s) case s: return #s
|
|
|
|
switch( state ) {
|
|
|
|
CASE_STR(COMMS_RELAYSTATE_UNCONNECTED);
|
2009-12-04 08:14:03 +00:00
|
|
|
CASE_STR(COMMS_RELAYSTATE_DENIED);
|
2009-08-21 12:00:09 +00:00
|
|
|
CASE_STR(COMMS_RELAYSTATE_CONNECT_PENDING);
|
|
|
|
CASE_STR(COMMS_RELAYSTATE_CONNECTED);
|
2009-11-08 21:30:13 +00:00
|
|
|
CASE_STR(COMMS_RELAYSTATE_RECONNECTED);
|
2009-08-21 12:00:09 +00:00
|
|
|
CASE_STR(COMMS_RELAYSTATE_ALLCONNECTED);
|
2017-11-10 21:34:02 -08:00
|
|
|
#ifdef RELAY_VIA_HTTP
|
|
|
|
CASE_STR(COMMS_RELAYSTATE_USING_HTTP);
|
|
|
|
#endif
|
2009-08-21 12:00:09 +00:00
|
|
|
default:
|
2009-08-29 13:06:48 +00:00
|
|
|
XP_ASSERT(0);
|
2009-08-21 12:00:09 +00:00
|
|
|
}
|
|
|
|
#undef CASE_STR
|
|
|
|
return NULL;
|
|
|
|
}
|
2010-11-18 22:27:06 -08:00
|
|
|
|
|
|
|
const char*
|
|
|
|
XWREASON2Str( XWREASON reason )
|
|
|
|
{
|
|
|
|
#define CASE_STR(s) case s: return #s
|
|
|
|
switch( reason ) {
|
|
|
|
CASE_STR(XWRELAY_ERROR_NONE);
|
|
|
|
CASE_STR(XWRELAY_ERROR_OLDFLAGS);
|
|
|
|
CASE_STR(XWRELAY_ERROR_BADPROTO);
|
|
|
|
CASE_STR(XWRELAY_ERROR_RELAYBUSY);
|
|
|
|
CASE_STR(XWRELAY_ERROR_SHUTDOWN);
|
|
|
|
CASE_STR(XWRELAY_ERROR_TIMEOUT);
|
|
|
|
CASE_STR(XWRELAY_ERROR_HEART_YOU);
|
|
|
|
CASE_STR(XWRELAY_ERROR_HEART_OTHER);
|
|
|
|
CASE_STR(XWRELAY_ERROR_LOST_OTHER);
|
|
|
|
CASE_STR(XWRELAY_ERROR_OTHER_DISCON);
|
|
|
|
CASE_STR(XWRELAY_ERROR_NO_ROOM);
|
|
|
|
CASE_STR(XWRELAY_ERROR_DUP_ROOM);
|
|
|
|
CASE_STR(XWRELAY_ERROR_TOO_MANY);
|
|
|
|
CASE_STR(XWRELAY_ERROR_DELETED);
|
2011-06-23 07:12:50 -07:00
|
|
|
CASE_STR(XWRELAY_ERROR_NORECONN);
|
2011-07-06 06:47:56 -07:00
|
|
|
CASE_STR(XWRELAY_ERROR_DEADGAME);
|
2010-11-18 22:27:06 -08:00
|
|
|
CASE_STR(XWRELAY_ERROR_LASTERR);
|
|
|
|
default:
|
|
|
|
XP_ASSERT(0);
|
|
|
|
}
|
|
|
|
#undef CASE_STR
|
|
|
|
return NULL;
|
|
|
|
}
|
2009-08-21 12:00:09 +00:00
|
|
|
#endif
|
|
|
|
|
|
|
|
static void
|
2020-04-25 17:47:07 -07:00
|
|
|
set_relay_state( CommsCtxt* comms, XWEnv xwe, CommsRelayState state )
|
2009-08-21 12:00:09 +00:00
|
|
|
{
|
2014-02-07 21:12:45 -08:00
|
|
|
if ( comms->rr.relayState != state ) {
|
2021-12-15 10:03:09 -08:00
|
|
|
XP_LOGFF( TAGFMT() "%s => %s", TAGPRMS,
|
|
|
|
CommsRelayState2Str(comms->rr.relayState),
|
|
|
|
CommsRelayState2Str(state) );
|
2014-02-07 21:12:45 -08:00
|
|
|
comms->rr.relayState = state;
|
2009-09-12 21:39:13 +00:00
|
|
|
if ( !!comms->procs.rstatus ) {
|
2020-04-25 17:47:07 -07:00
|
|
|
(*comms->procs.rstatus)( xwe, comms->procs.closure, state );
|
2009-09-12 21:39:13 +00:00
|
|
|
}
|
2009-08-21 12:00:09 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2009-02-28 19:27:45 +00:00
|
|
|
static void
|
2020-04-25 17:47:07 -07:00
|
|
|
init_relay( CommsCtxt* comms, XWEnv xwe, XP_U16 nPlayersHere, XP_U16 nPlayersTotal )
|
2009-02-28 19:27:45 +00:00
|
|
|
{
|
2014-02-07 21:12:45 -08:00
|
|
|
comms->rr.myHostID = comms->isServer? HOST_ID_SERVER: HOST_ID_NONE;
|
|
|
|
if ( HOST_ID_NONE != comms->rr.myHostID ) {
|
2022-04-29 09:19:44 -07:00
|
|
|
XP_LOGFF( "set hostid: %x", comms->rr.myHostID );
|
2013-07-01 07:54:38 -07:00
|
|
|
}
|
2020-04-25 17:47:07 -07:00
|
|
|
set_relay_state( comms, xwe, COMMS_RELAYSTATE_UNCONNECTED );
|
2014-02-07 21:12:45 -08:00
|
|
|
comms->rr.nPlayersHere = nPlayersHere;
|
|
|
|
comms->rr.nPlayersTotal = nPlayersTotal;
|
|
|
|
comms->rr.cookieID = COOKIE_ID_NONE;
|
|
|
|
comms->rr.connName[0] = '\0';
|
2009-02-28 19:27:45 +00:00
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2003-11-01 05:35:29 +00:00
|
|
|
CommsCtxt*
|
2020-04-25 17:47:07 -07:00
|
|
|
comms_make( MPFORMAL XWEnv xwe, XW_UtilCtxt* util, XP_Bool isServer,
|
2022-09-03 18:47:04 -07:00
|
|
|
const CommsAddrRec* selfAddr, const CommsAddrRec* hostAddr,
|
2023-12-04 09:31:11 -08:00
|
|
|
const TransportProcs* procs,
|
2022-05-14 07:10:48 -07:00
|
|
|
#ifdef XWFEATURE_RELAY
|
|
|
|
XP_U16 nPlayersHere, XP_U16 nPlayersTotal,
|
2020-08-25 14:47:56 -07:00
|
|
|
RoleChangeProc rcp, void* rcClosure,
|
2023-12-04 09:31:11 -08:00
|
|
|
#endif
|
2020-08-25 14:47:56 -07:00
|
|
|
XP_U16 forceChannel
|
2011-07-31 22:23:46 -07:00
|
|
|
#ifdef SET_GAMESEED
|
|
|
|
, XP_U16 gameSeed
|
|
|
|
#endif
|
|
|
|
)
|
2003-11-01 05:35:29 +00:00
|
|
|
{
|
2018-07-05 07:58:50 -07:00
|
|
|
CommsCtxt* comms = (CommsCtxt*)XP_CALLOC( mpool, sizeof(*comms) );
|
2015-02-09 19:15:43 -08:00
|
|
|
#ifdef DEBUG
|
|
|
|
comms->tag = mpool_getTag(mpool);
|
2021-12-15 10:03:09 -08:00
|
|
|
XP_LOGFF( TAGFMT(isServer=%d; forceChannel=%d), TAGPRMS, isServer, forceChannel );
|
2015-02-09 19:15:43 -08:00
|
|
|
#endif
|
|
|
|
MPASSIGN(comms->mpool, mpool);
|
2003-11-01 05:35:29 +00:00
|
|
|
|
2014-12-21 17:40:00 -08:00
|
|
|
XP_ASSERT( 0 == (forceChannel & ~CHANNEL_MASK) );
|
2015-02-09 19:15:43 -08:00
|
|
|
comms->isServer = isServer;
|
|
|
|
comms->forceChannel = forceChannel;
|
2011-01-23 21:45:41 -08:00
|
|
|
if ( !!procs ) {
|
2015-02-09 19:15:43 -08:00
|
|
|
XP_MEMCPY( &comms->procs, procs, sizeof(comms->procs) );
|
2011-08-16 19:36:23 -07:00
|
|
|
#ifdef COMMS_XPORT_FLAGSPROC
|
2020-04-25 17:47:07 -07:00
|
|
|
comms->xportFlags = (*comms->procs.getFlags)(xwe, comms->procs.closure);
|
2011-08-16 19:36:23 -07:00
|
|
|
#else
|
2015-02-09 19:15:43 -08:00
|
|
|
comms->xportFlags = comms->procs.flags;
|
2011-08-16 19:36:23 -07:00
|
|
|
#endif
|
2011-01-23 21:45:41 -08:00
|
|
|
}
|
2015-02-09 19:15:43 -08:00
|
|
|
comms->util = util;
|
2020-04-26 11:28:18 -07:00
|
|
|
comms->dutil = util_getDevUtilCtxt( util, xwe );
|
2003-11-01 05:35:29 +00:00
|
|
|
|
2006-10-10 01:34:37 +00:00
|
|
|
#ifdef XWFEATURE_RELAY
|
2023-12-04 09:31:11 -08:00
|
|
|
XP_ASSERT( rcp );
|
|
|
|
comms->rcProc = rcp;
|
|
|
|
comms->rcClosure = rcClosure;
|
|
|
|
|
2020-04-25 17:47:07 -07:00
|
|
|
init_relay( comms, xwe, nPlayersHere, nPlayersTotal );
|
2011-07-31 22:23:46 -07:00
|
|
|
# ifdef SET_GAMESEED
|
2015-02-09 19:15:43 -08:00
|
|
|
comms->channelSeed = gameSeed;
|
2011-07-31 22:23:46 -07:00
|
|
|
# endif
|
2006-04-07 03:08:23 +00:00
|
|
|
#endif
|
2022-09-03 18:47:04 -07:00
|
|
|
|
|
|
|
if ( !!selfAddr ) {
|
2023-01-15 16:18:26 -08:00
|
|
|
ASSERT_ADDR_OK(selfAddr);
|
2023-02-16 21:36:46 -08:00
|
|
|
logAddrComms( comms, &comms->selfAddr, "before selfAddr" );
|
2023-01-15 16:18:26 -08:00
|
|
|
comms->selfAddr = *selfAddr;
|
2023-02-16 21:36:46 -08:00
|
|
|
logAddrComms( comms, &comms->selfAddr, "after selfAddr" );
|
2022-09-03 18:47:04 -07:00
|
|
|
}
|
|
|
|
if ( !!hostAddr ) {
|
|
|
|
XP_ASSERT( !isServer );
|
2023-02-16 21:36:46 -08:00
|
|
|
logAddrComms( comms, hostAddr, __func__ );
|
2022-09-03 18:47:04 -07:00
|
|
|
XP_PlayerAddr channelNo = comms_getChannelSeed( comms );
|
|
|
|
#ifdef DEBUG
|
|
|
|
AddressRecord* rec =
|
|
|
|
#endif
|
2023-02-15 08:31:51 -08:00
|
|
|
rememberChannelAddress( comms, channelNo, 0, hostAddr,
|
2022-12-21 16:11:55 -08:00
|
|
|
COMMS_VERSION );
|
2023-02-15 08:31:51 -08:00
|
|
|
XP_ASSERT( rec == getRecordFor( comms, channelNo ) );
|
2023-01-15 16:18:26 -08:00
|
|
|
#ifdef DEBUG
|
|
|
|
/* Anything in hostAddr should be supported -- in selfAddr */
|
|
|
|
CommsConnType typ;
|
|
|
|
for ( XP_U32 st = 0; addr_iter( hostAddr, &typ, &st ); ) {
|
2023-01-17 18:01:30 -08:00
|
|
|
if ( !addr_hasType( &comms->selfAddr, typ ) ) {
|
|
|
|
XP_LOGFF( "%s not in selfAddr", ConnType2Str(typ) );
|
2023-01-24 11:38:11 -08:00
|
|
|
/* PENDING: fix this */
|
|
|
|
// XP_ASSERT(0); <-- happening a lot (NFC missing)
|
2023-01-17 18:01:30 -08:00
|
|
|
}
|
2023-01-15 16:18:26 -08:00
|
|
|
}
|
|
|
|
#endif
|
2022-09-03 18:47:04 -07:00
|
|
|
}
|
|
|
|
|
2015-02-09 19:15:43 -08:00
|
|
|
return comms;
|
2003-11-01 05:35:29 +00:00
|
|
|
} /* comms_make */
|
|
|
|
|
|
|
|
static void
|
2023-02-01 19:08:04 -08:00
|
|
|
forEachElem( CommsCtxt* comms, EachMsgProc proc, void* closure )
|
2023-03-17 09:58:18 -07:00
|
|
|
|
2003-11-01 05:35:29 +00:00
|
|
|
{
|
2023-02-01 19:08:04 -08:00
|
|
|
THREAD_CHECK_START(comms);
|
2023-03-16 07:41:53 -07:00
|
|
|
for ( AddressRecord* recs = comms->recs; !!recs; recs = recs->next ) {
|
|
|
|
for ( MsgQueueElem** home = &recs->_msgQueueHead; !!*home; ) {
|
2023-03-17 09:58:18 -07:00
|
|
|
MsgQueueElem* elem = *home;
|
|
|
|
ForEachAct fea = (*proc)( elem, closure );
|
|
|
|
if ( 0 != (FEA_REMOVE & fea) ) {
|
2023-03-18 15:15:34 -07:00
|
|
|
*home = (MsgQueueElem*)elem->smp.next;
|
2023-03-17 09:58:18 -07:00
|
|
|
#ifdef DEBUG
|
2023-03-18 15:15:34 -07:00
|
|
|
elem->smp.next = NULL;
|
2023-03-17 09:58:18 -07:00
|
|
|
#endif
|
|
|
|
freeElem( MPPARM(comms->mpool) elem );
|
|
|
|
XP_ASSERT( 1 <= comms->queueLen );
|
|
|
|
--comms->queueLen;
|
|
|
|
} else {
|
2023-03-18 15:15:34 -07:00
|
|
|
home = (MsgQueueElem**)&elem->smp.next;
|
2023-03-17 09:58:18 -07:00
|
|
|
}
|
|
|
|
if ( 0 != (FEA_EXIT & fea) ) {
|
2023-03-16 07:41:53 -07:00
|
|
|
goto done;
|
|
|
|
}
|
2023-02-01 19:08:04 -08:00
|
|
|
}
|
2023-03-16 07:41:53 -07:00
|
|
|
}
|
|
|
|
done:
|
2023-02-01 19:08:04 -08:00
|
|
|
assertQueueOk( comms );
|
|
|
|
THREAD_CHECK_END();
|
|
|
|
}
|
|
|
|
|
|
|
|
static ForEachAct
|
|
|
|
freeElemProc( MsgQueueElem* XP_UNUSED(elem), void* XP_UNUSED(closure) )
|
|
|
|
{
|
|
|
|
return FEA_REMOVE;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
cleanupInternal( CommsCtxt* comms )
|
|
|
|
{
|
|
|
|
forEachElem( comms, freeElemProc, NULL );
|
|
|
|
XP_ASSERT( 0 == comms->queueLen );
|
2003-11-01 05:35:29 +00:00
|
|
|
} /* cleanupInternal */
|
|
|
|
|
|
|
|
static void
|
|
|
|
cleanupAddrRecs( CommsCtxt* comms )
|
|
|
|
{
|
|
|
|
AddressRecord* recs;
|
|
|
|
AddressRecord* next;
|
|
|
|
|
|
|
|
for ( recs = comms->recs; !!recs; recs = next ) {
|
|
|
|
next = recs->next;
|
2023-03-16 07:41:53 -07:00
|
|
|
XP_ASSERT( !recs->_msgQueueHead );
|
2003-11-01 05:35:29 +00:00
|
|
|
XP_FREE( comms->mpool, recs );
|
|
|
|
}
|
|
|
|
comms->recs = (AddressRecord*)NULL;
|
|
|
|
} /* cleanupAddrRecs */
|
|
|
|
|
2006-10-10 01:34:37 +00:00
|
|
|
#ifdef XWFEATURE_RELAY
|
2009-02-01 15:50:58 +00:00
|
|
|
static XP_Bool
|
2020-04-25 17:47:07 -07:00
|
|
|
p_comms_resetTimer( void* closure, XWEnv xwe, XWTimerReason XP_UNUSED_DBG(why) )
|
2009-02-01 15:50:58 +00:00
|
|
|
{
|
|
|
|
CommsCtxt* comms = (CommsCtxt*)closure;
|
|
|
|
LOG_FUNC();
|
|
|
|
XP_ASSERT( why == TIMER_COMMS );
|
|
|
|
|
2009-12-04 08:14:03 +00:00
|
|
|
/* Once we're denied we don't try again. A new game or save and re-open
|
|
|
|
will reset comms and get us out of this state. */
|
2014-02-07 21:12:45 -08:00
|
|
|
if ( comms->rr.relayState != COMMS_RELAYSTATE_DENIED ) {
|
|
|
|
XP_Bool success = comms->rr.relayState >= COMMS_RELAYSTATE_CONNECTED
|
2020-04-25 17:47:07 -07:00
|
|
|
|| relayConnect( comms, xwe );
|
2009-12-04 08:14:03 +00:00
|
|
|
|
|
|
|
if ( success ) {
|
|
|
|
comms->reconTimerPending = XP_FALSE;
|
|
|
|
setHeartbeatTimer( comms ); /* in case we killed it with this
|
|
|
|
one.... */
|
|
|
|
} else {
|
2020-04-26 11:28:18 -07:00
|
|
|
set_reset_timer( comms, xwe );
|
2009-12-04 08:14:03 +00:00
|
|
|
}
|
2009-02-01 15:50:58 +00:00
|
|
|
}
|
|
|
|
|
2009-12-04 08:14:03 +00:00
|
|
|
return XP_FALSE; /* no redraw required */
|
2009-02-01 15:50:58 +00:00
|
|
|
} /* p_comms_resetTimer */
|
|
|
|
|
|
|
|
static void
|
2020-04-26 11:28:18 -07:00
|
|
|
set_reset_timer( CommsCtxt* comms, XWEnv xwe )
|
2009-02-01 15:50:58 +00:00
|
|
|
{
|
|
|
|
/* This timer is allowed to overwrite a heartbeat timer, but not
|
|
|
|
vice-versa. Make sure we can restart it. */
|
|
|
|
comms->hbTimerPending = XP_FALSE;
|
2020-04-26 11:28:18 -07:00
|
|
|
util_setTimer( comms->util, xwe, TIMER_COMMS, 15,
|
2009-02-01 15:50:58 +00:00
|
|
|
p_comms_resetTimer, comms );
|
|
|
|
comms->reconTimerPending = XP_TRUE;
|
2009-09-22 03:50:37 +00:00
|
|
|
} /* set_reset_timer */
|
2022-04-28 07:37:33 -07:00
|
|
|
#endif /* XWFEATURE_RELAY */
|
2009-02-01 15:50:58 +00:00
|
|
|
|
2003-11-01 05:35:29 +00:00
|
|
|
void
|
2020-04-26 11:28:18 -07:00
|
|
|
comms_destroy( CommsCtxt* comms, XWEnv xwe )
|
2003-11-01 05:35:29 +00:00
|
|
|
{
|
2013-01-16 06:31:09 -08:00
|
|
|
/* did I call comms_stop()? */
|
2022-09-03 19:14:40 -07:00
|
|
|
XP_ASSERT( ! addr_hasType( &comms->selfAddr, COMMS_CONN_RELAY )
|
2014-02-07 21:12:45 -08:00
|
|
|
|| COMMS_RELAYSTATE_UNCONNECTED == comms->rr.relayState );
|
2013-01-16 06:31:09 -08:00
|
|
|
|
2003-11-01 05:35:29 +00:00
|
|
|
cleanupInternal( comms );
|
|
|
|
cleanupAddrRecs( comms );
|
|
|
|
|
2020-04-26 11:28:18 -07:00
|
|
|
util_clearTimer( comms->util, xwe, TIMER_COMMS );
|
2009-09-21 12:49:08 +00:00
|
|
|
|
2003-11-01 05:35:29 +00:00
|
|
|
XP_FREE( comms->mpool, comms );
|
|
|
|
} /* comms_destroy */
|
|
|
|
|
|
|
|
void
|
2023-02-05 08:59:37 -08:00
|
|
|
comms_setConnID( CommsCtxt* comms, XP_U32 connID, XP_U16 streamVersion )
|
2003-11-01 05:35:29 +00:00
|
|
|
{
|
2023-01-06 07:52:41 -08:00
|
|
|
THREAD_CHECK_START(comms);
|
2009-02-07 18:14:28 +00:00
|
|
|
XP_ASSERT( CONN_ID_NONE != connID );
|
2012-01-25 18:27:37 -08:00
|
|
|
XP_ASSERT( 0 == comms->connID || connID == comms->connID );
|
2003-11-01 05:35:29 +00:00
|
|
|
comms->connID = connID;
|
2023-02-05 08:59:37 -08:00
|
|
|
XP_ASSERT( 0 == comms->streamVersion
|
|
|
|
|| streamVersion == comms->streamVersion );
|
|
|
|
comms->streamVersion = streamVersion;
|
2023-12-18 17:08:51 -08:00
|
|
|
XP_LOGFF( "set connID (gameID) to %X, streamVersion to 0x%X",
|
2023-02-11 07:48:31 -08:00
|
|
|
connID, streamVersion );
|
2023-01-06 07:52:41 -08:00
|
|
|
THREAD_CHECK_END();
|
2003-11-01 05:35:29 +00:00
|
|
|
} /* comms_setConnID */
|
|
|
|
|
2005-03-19 22:01:38 +00:00
|
|
|
static void
|
2014-09-25 20:05:37 -07:00
|
|
|
addrFromStreamOne( CommsAddrRec* addrP, XWStreamCtxt* stream, CommsConnType typ )
|
2005-03-19 22:01:38 +00:00
|
|
|
{
|
2015-07-06 20:28:16 -07:00
|
|
|
XP_U16 version = stream_getVersion( stream );
|
2014-09-25 20:05:37 -07:00
|
|
|
switch( typ ) {
|
2007-12-07 07:50:19 +00:00
|
|
|
case COMMS_CONN_NONE:
|
2005-03-19 22:01:38 +00:00
|
|
|
break;
|
|
|
|
case COMMS_CONN_BT:
|
2014-09-25 20:05:37 -07:00
|
|
|
stringFromStreamHere( stream, addrP->u.bt.hostName,
|
|
|
|
sizeof(addrP->u.bt.hostName) );
|
|
|
|
stringFromStreamHere( stream, addrP->u.bt.btAddr.chars,
|
|
|
|
sizeof(addrP->u.bt.btAddr.chars) );
|
2006-09-08 07:14:24 +00:00
|
|
|
break;
|
2005-03-19 22:01:38 +00:00
|
|
|
case COMMS_CONN_IR:
|
|
|
|
/* nothing to save */
|
|
|
|
break;
|
2007-11-26 02:58:25 +00:00
|
|
|
case COMMS_CONN_IP_DIRECT:
|
2014-09-25 20:05:37 -07:00
|
|
|
stringFromStreamHere( stream, addrP->u.ip.hostName_ip,
|
|
|
|
sizeof(addrP->u.ip.hostName_ip) );
|
|
|
|
addrP->u.ip.ipAddr_ip = stream_getU32( stream );
|
|
|
|
addrP->u.ip.port_ip = stream_getU16( stream );
|
2005-03-19 22:01:38 +00:00
|
|
|
break;
|
2022-09-08 09:57:11 -07:00
|
|
|
case COMMS_CONN_RELAY: {
|
|
|
|
IpRelay ip_relay = {{0}};
|
2022-10-19 20:33:30 -07:00
|
|
|
if ( version < STREAM_VERS_NORELAY ) {
|
|
|
|
stringFromStreamHere( stream, ip_relay.invite,
|
|
|
|
sizeof(ip_relay.invite) );
|
|
|
|
stringFromStreamHere( stream, ip_relay.hostName,
|
|
|
|
sizeof(ip_relay.hostName) );
|
|
|
|
ip_relay.ipAddr = stream_getU32( stream );
|
|
|
|
ip_relay.port = stream_getU16( stream );
|
|
|
|
if ( version >= STREAM_VERS_DICTLANG ) {
|
|
|
|
ip_relay.seeksPublicRoom = stream_getBits( stream, 1 );
|
|
|
|
ip_relay.advertiseRoom = stream_getBits( stream, 1 );
|
|
|
|
}
|
2022-09-08 09:57:11 -07:00
|
|
|
#ifdef XWFEATURE_RELAY
|
2022-10-19 20:33:30 -07:00
|
|
|
XP_MEMCPY( &addrP->u.ip_relay, &ip_relay, sizeof(ip_relay) );
|
2022-09-08 09:57:11 -07:00
|
|
|
#endif
|
2022-10-19 20:33:30 -07:00
|
|
|
}
|
2022-09-08 09:57:11 -07:00
|
|
|
}
|
2005-03-19 22:01:38 +00:00
|
|
|
break;
|
2008-12-29 01:35:29 +00:00
|
|
|
case COMMS_CONN_SMS:
|
2014-09-25 20:05:37 -07:00
|
|
|
stringFromStreamHere( stream, addrP->u.sms.phone,
|
|
|
|
sizeof(addrP->u.sms.phone) );
|
|
|
|
addrP->u.sms.port = stream_getU16( stream );
|
2008-12-29 01:35:29 +00:00
|
|
|
break;
|
2016-11-14 08:06:53 -08:00
|
|
|
case COMMS_CONN_P2P:
|
|
|
|
stringFromStreamHere( stream, addrP->u.p2p.mac_addr,
|
|
|
|
sizeof(addrP->u.p2p.mac_addr) );
|
|
|
|
break;
|
2019-11-23 10:53:51 -08:00
|
|
|
case COMMS_CONN_NFC:
|
|
|
|
break;
|
2020-05-20 13:58:53 -07:00
|
|
|
case COMMS_CONN_MQTT:
|
|
|
|
stream_getBytes( stream, &addrP->u.mqtt.devID, sizeof(addrP->u.mqtt.devID) );
|
|
|
|
break;
|
2005-03-19 22:01:38 +00:00
|
|
|
default:
|
|
|
|
/* shut up, compiler */
|
|
|
|
break;
|
|
|
|
}
|
2014-09-25 20:05:37 -07:00
|
|
|
} /* addrFromStreamOne */
|
2005-03-19 22:01:38 +00:00
|
|
|
|
2014-11-02 17:24:37 -08:00
|
|
|
void
|
2014-09-25 20:05:37 -07:00
|
|
|
addrFromStream( CommsAddrRec* addrP, XWStreamCtxt* stream )
|
|
|
|
{
|
2023-12-18 17:08:51 -08:00
|
|
|
XP_MEMSET( addrP, 0, sizeof(*addrP) );
|
2014-09-25 20:05:37 -07:00
|
|
|
XP_U8 tmp = stream_getU8( stream );
|
2023-02-16 21:36:46 -08:00
|
|
|
XP_U16 version = stream_getVersion( stream );
|
|
|
|
XP_ASSERT( 0 < version );
|
|
|
|
if ( STREAM_VERS_MULTIADDR > version && (COMMS_CONN_NONE != tmp) ) {
|
2014-09-25 22:24:09 -07:00
|
|
|
tmp = 1 << (tmp - 1);
|
2014-09-25 20:05:37 -07:00
|
|
|
}
|
|
|
|
addrP->_conTypes = tmp;
|
|
|
|
|
|
|
|
CommsConnType typ;
|
2014-09-25 22:24:09 -07:00
|
|
|
for ( XP_U32 st = 0; addr_iter( addrP, &typ, &st ); ) {
|
2014-09-25 20:05:37 -07:00
|
|
|
addrFromStreamOne( addrP, stream, typ );
|
|
|
|
}
|
2022-09-04 09:59:47 -07:00
|
|
|
// ASSERT_ADDR_OK( addrP );
|
2014-09-25 20:05:37 -07:00
|
|
|
}
|
2005-03-19 22:01:38 +00:00
|
|
|
|
2022-02-10 08:28:34 -08:00
|
|
|
/* Return TRUE if there are no addresses left that include relay */
|
|
|
|
static XP_Bool
|
2022-02-12 13:27:24 -08:00
|
|
|
removeRelayIf( CommsCtxt* comms, XWEnv xwe )
|
2022-02-10 08:28:34 -08:00
|
|
|
{
|
|
|
|
XP_Bool allRemoved = XP_TRUE;
|
2022-02-12 13:27:24 -08:00
|
|
|
XP_UCHAR bufs[MAX_NUM_PLAYERS+1][64];
|
|
|
|
const XP_UCHAR* ptrs[MAX_NUM_PLAYERS+1];
|
|
|
|
int nIds = 0;
|
2022-02-10 08:28:34 -08:00
|
|
|
for ( AddressRecord* rec = comms->recs; !!rec; rec = rec->next ) {
|
|
|
|
CommsAddrRec* addr = &rec->addr;
|
|
|
|
if ( addr_hasType( addr, COMMS_CONN_RELAY ) ) {
|
|
|
|
if ( addr_hasType( addr, COMMS_CONN_MQTT )
|
|
|
|
&& 0 != addr->u.mqtt.devID ) {
|
|
|
|
addr_rmType( addr, COMMS_CONN_RELAY );
|
|
|
|
} else {
|
2022-02-12 13:27:24 -08:00
|
|
|
XP_U16 len = VSIZE(bufs[nIds]);
|
|
|
|
if ( formatRelayID( comms, rec->rr.hostID, bufs[nIds], &len ) ) {
|
|
|
|
ptrs[nIds] = &bufs[nIds][0];
|
|
|
|
++nIds;
|
|
|
|
}
|
2022-02-10 08:28:34 -08:00
|
|
|
allRemoved = XP_FALSE;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2022-02-12 13:27:24 -08:00
|
|
|
if ( 0 < nIds ) {
|
|
|
|
util_getMQTTIDsFor( comms->util, xwe, nIds, ptrs );
|
|
|
|
}
|
2022-02-10 08:28:34 -08:00
|
|
|
LOG_RETURNF( "%s", boolToStr(allRemoved) );
|
|
|
|
return allRemoved;
|
2022-04-28 07:37:33 -07:00
|
|
|
} /* removeRelayIf */
|
2022-02-10 08:28:34 -08:00
|
|
|
|
2022-10-08 12:26:00 -07:00
|
|
|
/* Looking toward a time when we store only the first couple of bits of
|
|
|
|
channelNo. Not possible yet, though. */
|
|
|
|
static XP_U16
|
|
|
|
readChannelNo( XWStreamCtxt* stream )
|
|
|
|
{
|
|
|
|
XP_U16 tmp = stream_getU16( stream );
|
|
|
|
return tmp; // & CHANNEL_MASK;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
writeChannelNo( XWStreamCtxt* stream, XP_PlayerAddr channelNo )
|
|
|
|
{
|
|
|
|
// stream_putU16( stream, channelNo & CHANNEL_MASK );
|
|
|
|
stream_putU16( stream, channelNo );
|
|
|
|
}
|
|
|
|
|
2003-11-01 05:35:29 +00:00
|
|
|
CommsCtxt*
|
2020-08-25 14:47:56 -07:00
|
|
|
comms_makeFromStream( MPFORMAL XWEnv xwe, XWStreamCtxt* stream,
|
2020-08-25 19:38:35 -07:00
|
|
|
XW_UtilCtxt* util, XP_Bool isServer,
|
2020-08-25 14:47:56 -07:00
|
|
|
const TransportProcs* procs,
|
2023-12-04 09:31:11 -08:00
|
|
|
#ifdef XWFEATURE_RELAY
|
2020-08-25 14:47:56 -07:00
|
|
|
RoleChangeProc rcp, void* rcClosure,
|
2023-12-04 09:31:11 -08:00
|
|
|
#endif
|
2020-08-25 14:47:56 -07:00
|
|
|
XP_U16 forceChannel )
|
2003-11-01 05:35:29 +00:00
|
|
|
{
|
2022-10-07 09:23:09 -07:00
|
|
|
LOG_FUNC();
|
2006-10-05 01:17:03 +00:00
|
|
|
XP_U16 version = stream_getVersion( stream );
|
2003-11-01 05:35:29 +00:00
|
|
|
|
2020-09-26 10:12:29 -07:00
|
|
|
XP_U8 flags = stream_getU8( stream );
|
2020-09-25 08:58:33 -07:00
|
|
|
if ( version < STREAM_VERS_GICREATED ) {
|
|
|
|
flags = 0;
|
|
|
|
}
|
2022-01-08 20:38:29 -08:00
|
|
|
|
2023-01-17 18:01:30 -08:00
|
|
|
CommsAddrRec selfAddr = {0};
|
|
|
|
addrFromStream( &selfAddr, stream );
|
|
|
|
if ( addr_hasType( &selfAddr, COMMS_CONN_MQTT )
|
|
|
|
&& 0 == selfAddr.u.mqtt.devID ) {
|
2022-09-04 09:59:47 -07:00
|
|
|
XW_DUtilCtxt* dutil = util_getDevUtilCtxt( util, xwe );
|
2023-01-17 18:01:30 -08:00
|
|
|
dvc_getMQTTDevID( dutil, xwe, &selfAddr.u.mqtt.devID );
|
2022-09-04 09:59:47 -07:00
|
|
|
}
|
2023-01-17 18:01:30 -08:00
|
|
|
ASSERT_ADDR_OK( &selfAddr );
|
2006-10-05 01:17:03 +00:00
|
|
|
|
2022-01-08 20:38:29 -08:00
|
|
|
XP_U16 nPlayersHere, nPlayersTotal;
|
2015-07-06 20:28:16 -07:00
|
|
|
if ( version >= STREAM_VERS_DEVIDS
|
2023-01-17 18:01:30 -08:00
|
|
|
|| addr_hasType( &selfAddr, COMMS_CONN_RELAY ) ) {
|
2006-10-05 01:17:03 +00:00
|
|
|
nPlayersHere = (XP_U16)stream_getBits( stream, 4 );
|
|
|
|
nPlayersTotal = (XP_U16)stream_getBits( stream, 4 );
|
|
|
|
} else {
|
|
|
|
nPlayersHere = 0;
|
|
|
|
nPlayersTotal = 0;
|
|
|
|
}
|
2023-12-04 09:31:11 -08:00
|
|
|
CommsCtxt* comms = comms_make( MPPARM(mpool) xwe, util, isServer,
|
|
|
|
NULL, NULL, procs,
|
2022-05-14 07:10:48 -07:00
|
|
|
#ifdef XWFEATURE_RELAY
|
2023-12-04 09:31:11 -08:00
|
|
|
nPlayersHere, nPlayersTotal, rcp, rcClosure,
|
2022-05-14 07:10:48 -07:00
|
|
|
#endif
|
2023-12-04 09:31:11 -08:00
|
|
|
forceChannel
|
2011-07-31 22:23:46 -07:00
|
|
|
#ifdef SET_GAMESEED
|
2014-11-05 07:49:17 -08:00
|
|
|
, 0
|
2011-07-31 22:23:46 -07:00
|
|
|
#endif
|
2014-11-05 07:49:17 -08:00
|
|
|
);
|
2022-05-14 07:10:48 -07:00
|
|
|
#ifndef XWFEATURE_RELAY
|
|
|
|
XP_USE( nPlayersHere );
|
|
|
|
XP_USE( nPlayersTotal );
|
|
|
|
#endif
|
2023-01-17 18:01:30 -08:00
|
|
|
XP_MEMCPY( &comms->selfAddr, &selfAddr, sizeof(comms->selfAddr) );
|
2023-02-16 21:36:46 -08:00
|
|
|
logAddrComms( comms, &selfAddr, __func__ );
|
2020-09-21 11:31:49 -07:00
|
|
|
comms->flags = flags;
|
2003-11-01 05:35:29 +00:00
|
|
|
|
|
|
|
comms->connID = stream_getU32( stream );
|
2023-02-05 08:59:37 -08:00
|
|
|
XP_ASSERT( comms->streamVersion == 0 );
|
|
|
|
if ( version >= STREAM_VERS_MSGSTREAMVERS ) {
|
|
|
|
comms->streamVersion = stream_getU16( stream );
|
|
|
|
}
|
|
|
|
|
2022-10-08 12:26:00 -07:00
|
|
|
comms->nextChannelNo = readChannelNo( stream );
|
2009-09-26 14:37:49 +00:00
|
|
|
if ( version < STREAM_VERS_CHANNELSEED ) {
|
|
|
|
comms->channelSeed = 0;
|
|
|
|
} else {
|
2015-02-23 18:28:31 -08:00
|
|
|
comms->channelSeed = stream_getU16( stream );
|
2009-09-26 14:37:49 +00:00
|
|
|
}
|
2012-11-13 21:25:03 -08:00
|
|
|
if ( STREAM_VERS_COMMSBACKOFF <= version ) {
|
|
|
|
comms->resendBackoff = stream_getU16( stream );
|
|
|
|
comms->nextResend = stream_getU32( stream );
|
|
|
|
}
|
2023-01-17 18:01:30 -08:00
|
|
|
if ( addr_hasType( &selfAddr, COMMS_CONN_RELAY ) ) {
|
2014-02-07 21:12:45 -08:00
|
|
|
comms->rr.myHostID = stream_getU8( stream );
|
2020-08-24 16:16:20 -07:00
|
|
|
XP_LOGFF( "loaded myHostID: %d", comms->rr.myHostID );
|
2014-02-07 21:12:45 -08:00
|
|
|
stringFromStreamHere( stream, comms->rr.connName,
|
|
|
|
sizeof(comms->rr.connName) );
|
2006-10-05 01:17:03 +00:00
|
|
|
}
|
2005-03-19 22:01:38 +00:00
|
|
|
|
2023-02-01 19:08:04 -08:00
|
|
|
XP_U16 queueLen = stream_getU8( stream );
|
2003-11-01 05:35:29 +00:00
|
|
|
|
2020-01-24 09:05:16 -08:00
|
|
|
XP_U16 nAddrRecs = stream_getU8( stream );
|
2022-10-07 09:23:09 -07:00
|
|
|
XP_LOGFF( "nAddrRecs: %d", nAddrRecs );
|
2022-01-08 20:38:29 -08:00
|
|
|
AddressRecord** prevsAddrNext = &comms->recs;
|
2021-12-21 16:23:24 -08:00
|
|
|
for ( int ii = 0; ii < nAddrRecs; ++ii ) {
|
2012-04-13 20:21:59 -07:00
|
|
|
AddressRecord* rec = (AddressRecord*)XP_CALLOC( mpool, sizeof(*rec));
|
2004-10-02 03:47:24 +00:00
|
|
|
|
2006-10-05 01:17:03 +00:00
|
|
|
addrFromStream( &rec->addr, stream );
|
2023-02-16 21:36:46 -08:00
|
|
|
logAddrComms( comms, &rec->addr, __func__ );
|
2004-10-08 23:54:57 +00:00
|
|
|
|
2021-12-21 16:23:24 -08:00
|
|
|
if ( STREAM_VERS_SMALLCOMMS <= version ) {
|
2022-01-04 11:12:58 -08:00
|
|
|
rec->nextMsgID = stream_getU32VL( stream );
|
|
|
|
rec->lastMsgSaved = rec->lastMsgRcd = stream_getU32VL( stream );
|
2021-12-21 16:23:24 -08:00
|
|
|
rec->flags = stream_getU16( stream );
|
2022-01-04 11:12:58 -08:00
|
|
|
} else {
|
|
|
|
rec->nextMsgID = stream_getU16( stream );
|
|
|
|
rec->lastMsgSaved = rec->lastMsgRcd = stream_getU16( stream );
|
2021-12-21 16:23:24 -08:00
|
|
|
}
|
2019-11-01 07:22:56 -07:00
|
|
|
#ifdef LOG_COMMS_MSGNOS
|
2022-04-29 09:19:44 -07:00
|
|
|
XP_LOGFF( "read lastMsgRcd of %d for addr %d", rec->lastMsgRcd, ii );
|
2019-11-01 07:22:56 -07:00
|
|
|
#endif
|
2012-04-13 20:57:35 -07:00
|
|
|
if ( version >= STREAM_VERS_BLUETOOTH2 ) {
|
2012-04-13 20:21:59 -07:00
|
|
|
rec->lastMsgAckd = stream_getU16( stream );
|
|
|
|
}
|
2022-10-08 12:26:00 -07:00
|
|
|
rec->channelNo = readChannelNo( stream );
|
2014-10-15 07:26:18 -07:00
|
|
|
if ( addr_hasType( &rec->addr, COMMS_CONN_RELAY ) ) {
|
2014-02-07 21:12:45 -08:00
|
|
|
rec->rr.hostID = stream_getU8( stream );
|
2006-10-05 01:17:03 +00:00
|
|
|
}
|
2003-11-01 05:35:29 +00:00
|
|
|
|
2022-10-07 09:23:09 -07:00
|
|
|
CNO_FMT( cbuf, rec->channelNo );
|
|
|
|
XP_LOGFF( "loaded rec %d: %s", ii, cbuf );
|
|
|
|
|
2003-11-01 05:35:29 +00:00
|
|
|
*prevsAddrNext = rec;
|
|
|
|
prevsAddrNext = &rec->next;
|
|
|
|
}
|
|
|
|
|
2023-02-01 19:08:04 -08:00
|
|
|
for ( int ii = 0; ii < queueLen; ++ii ) {
|
2012-04-13 20:21:59 -07:00
|
|
|
MsgQueueElem* msg = (MsgQueueElem*)XP_CALLOC( mpool, sizeof(*msg) );
|
2003-11-01 05:35:29 +00:00
|
|
|
|
2022-10-08 12:26:00 -07:00
|
|
|
msg->channelNo = readChannelNo( stream );
|
2021-12-21 16:23:24 -08:00
|
|
|
if ( version >= STREAM_VERS_SMALLCOMMS ) {
|
2022-01-04 11:12:58 -08:00
|
|
|
msg->msgID = stream_getU32VL( stream );
|
2023-03-18 15:15:34 -07:00
|
|
|
msg->smp.len = stream_getU32VL( stream );
|
2021-12-21 16:23:24 -08:00
|
|
|
} else {
|
|
|
|
msg->msgID = stream_getU32( stream );
|
2023-03-18 15:15:34 -07:00
|
|
|
msg->smp.len = stream_getU16( stream );
|
2021-12-21 16:23:24 -08:00
|
|
|
}
|
2022-06-29 12:29:32 -07:00
|
|
|
if ( version >= STREAM_VERS_MSGTIMESTAMP ) {
|
2023-03-18 15:15:34 -07:00
|
|
|
msg->smp.createdStamp = stream_getU32( stream );
|
2022-06-29 12:29:32 -07:00
|
|
|
}
|
2023-03-18 15:15:34 -07:00
|
|
|
if ( 0 == msg->smp.createdStamp ) {
|
|
|
|
msg->smp.createdStamp = dutil_getCurSeconds( comms->dutil, xwe );
|
2022-12-29 22:39:02 -08:00
|
|
|
XP_LOGFF( "added missing timestamp" );
|
|
|
|
}
|
2012-11-13 21:18:52 -08:00
|
|
|
#ifdef DEBUG
|
2007-11-26 02:58:25 +00:00
|
|
|
msg->sendCount = 0;
|
|
|
|
#endif
|
2023-03-18 15:15:34 -07:00
|
|
|
XP_U16 len = msg->smp.len;
|
2022-09-06 12:16:29 -07:00
|
|
|
if ( 0 == len ) {
|
2022-12-27 16:03:08 -08:00
|
|
|
XP_ASSERT( isServer );
|
2022-09-06 12:16:29 -07:00
|
|
|
XP_U32 nliLen = stream_getU32VL( stream );
|
|
|
|
XWStreamCtxt* nliStream = mem_stream_make_raw( MPPARM(comms->mpool)
|
|
|
|
dutil_getVTManager(comms->dutil));
|
|
|
|
stream_getFromStream( nliStream, stream, nliLen );
|
|
|
|
NetLaunchInfo nli;
|
|
|
|
if ( nli_makeFromStream( &nli, nliStream ) ) {
|
2023-03-18 15:15:34 -07:00
|
|
|
msg->smp.buf = (XP_U8*)XP_MALLOC( mpool, sizeof(nli) );
|
|
|
|
XP_MEMCPY( (void*)msg->smp.buf, &nli, sizeof(nli) );
|
2022-09-06 12:16:29 -07:00
|
|
|
len = sizeof(nli); /* needed for checksum calc */
|
|
|
|
} else {
|
|
|
|
XP_ASSERT(0);
|
|
|
|
}
|
2023-02-03 22:00:47 -08:00
|
|
|
stream_destroy( nliStream );
|
2022-09-06 12:16:29 -07:00
|
|
|
} else {
|
2023-03-18 15:15:34 -07:00
|
|
|
msg->smp.buf = (XP_U8*)XP_MALLOC( mpool, len );
|
|
|
|
stream_getBytes( stream, (XP_U8*)msg->smp.buf, len );
|
2022-09-06 12:16:29 -07:00
|
|
|
}
|
2010-10-18 22:08:04 -07:00
|
|
|
#ifdef COMMS_CHECKSUM
|
2023-03-18 15:15:34 -07:00
|
|
|
msg->checksum = dutil_md5sum( comms->dutil, xwe, msg->smp.buf, len );
|
2010-10-18 20:17:44 -07:00
|
|
|
#endif
|
2023-03-18 15:15:34 -07:00
|
|
|
XP_ASSERT( NULL == msg->smp.next );
|
2023-04-07 19:53:51 -07:00
|
|
|
if ( !addToQueue( comms, xwe, msg, XP_FALSE ) ) {
|
|
|
|
--queueLen; /* was dropped */
|
|
|
|
}
|
2003-11-01 05:35:29 +00:00
|
|
|
}
|
2023-04-20 14:24:10 -07:00
|
|
|
#ifdef DEBUG
|
|
|
|
if ( queueLen != comms->queueLen ) {
|
|
|
|
XP_LOGFF( "Error: queueLen %d != comms->queueLen %d",
|
|
|
|
queueLen, comms->queueLen );
|
2023-10-26 12:25:59 -07:00
|
|
|
XP_ASSERT(0);
|
2023-04-20 14:24:10 -07:00
|
|
|
}
|
|
|
|
#endif
|
2003-11-01 05:35:29 +00:00
|
|
|
|
2017-05-26 07:30:42 -07:00
|
|
|
if ( STREAM_VERS_DISABLEDS <= version ) {
|
2019-11-30 18:39:11 -08:00
|
|
|
for ( CommsConnType typ = (CommsConnType)0; typ < VSIZE(comms->disableds); ++typ ) {
|
2022-09-03 19:14:40 -07:00
|
|
|
if ( typ < COMMS_CONN_NFC || addr_hasType( &comms->selfAddr, typ ) ) {
|
2019-11-30 18:39:11 -08:00
|
|
|
for ( int ii = 0; ii < VSIZE(comms->disableds[0]); ++ii ) {
|
|
|
|
comms->disableds[typ][ii] = 0 != stream_getBits( stream, 1 );
|
|
|
|
}
|
2017-05-26 07:30:42 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-04-25 17:47:07 -07:00
|
|
|
notifyQueueChanged( comms, xwe );
|
2022-09-03 19:14:40 -07:00
|
|
|
if ( addr_hasType( &comms->selfAddr, COMMS_CONN_RELAY )
|
2022-02-12 13:27:24 -08:00
|
|
|
&& removeRelayIf( comms, xwe ) ) {
|
2022-09-03 19:14:40 -07:00
|
|
|
addr_rmType( &comms->selfAddr, COMMS_CONN_RELAY );
|
2022-02-10 08:28:34 -08:00
|
|
|
}
|
2019-12-10 14:42:54 -08:00
|
|
|
|
2022-10-07 09:23:09 -07:00
|
|
|
listRecs( comms, __func__ );
|
|
|
|
|
|
|
|
LOG_RETURNF( "%p", comms );
|
2003-11-01 05:35:29 +00:00
|
|
|
return comms;
|
|
|
|
} /* comms_makeFromStream */
|
|
|
|
|
2007-11-26 02:58:25 +00:00
|
|
|
#ifdef COMMS_HEARTBEAT
|
2008-12-29 01:35:29 +00:00
|
|
|
static void
|
|
|
|
setDoHeartbeat( CommsCtxt* comms )
|
|
|
|
{
|
2022-09-03 19:14:40 -07:00
|
|
|
CommsConnType conType = comms->selfAddr.conType;
|
2008-12-29 01:35:29 +00:00
|
|
|
comms->doHeartbeat = XP_FALSE
|
|
|
|
|| COMMS_CONN_IP_DIRECT == conType
|
|
|
|
|| COMMS_CONN_BT == conType
|
|
|
|
;
|
|
|
|
}
|
|
|
|
#else
|
|
|
|
# define setDoHeartbeat(c)
|
2007-11-26 02:58:25 +00:00
|
|
|
#endif
|
|
|
|
|
2009-02-28 19:27:45 +00:00
|
|
|
/*
|
|
|
|
* Currently this disconnects an open connection. Don't do that.
|
|
|
|
*/
|
2008-12-29 01:35:29 +00:00
|
|
|
void
|
2020-04-25 17:47:07 -07:00
|
|
|
comms_start( CommsCtxt* comms, XWEnv xwe )
|
2008-12-29 01:35:29 +00:00
|
|
|
{
|
2009-09-21 12:49:08 +00:00
|
|
|
XP_ASSERT( !!comms );
|
2008-12-29 01:35:29 +00:00
|
|
|
setDoHeartbeat( comms );
|
2022-05-14 07:10:48 -07:00
|
|
|
sendConnect( comms, xwe
|
|
|
|
#ifdef XWFEATURE_RELAY
|
|
|
|
, XP_FALSE
|
|
|
|
#endif
|
|
|
|
);
|
2007-11-26 02:58:25 +00:00
|
|
|
} /* comms_start */
|
|
|
|
|
2013-01-16 06:31:09 -08:00
|
|
|
void
|
2022-05-14 07:10:48 -07:00
|
|
|
comms_stop( CommsCtxt* comms
|
|
|
|
#ifdef XWFEATURE_RELAY
|
|
|
|
, XWEnv xwe
|
|
|
|
#endif
|
|
|
|
)
|
2013-01-16 06:31:09 -08:00
|
|
|
{
|
2022-09-03 19:14:40 -07:00
|
|
|
if ( addr_hasType( &comms->selfAddr, COMMS_CONN_RELAY ) ) {
|
2020-04-25 17:47:07 -07:00
|
|
|
relayDisconnect( comms, xwe );
|
2013-01-16 06:31:09 -08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2007-11-26 02:58:25 +00:00
|
|
|
static void
|
2022-05-14 07:10:48 -07:00
|
|
|
sendConnect( CommsCtxt* comms, XWEnv xwe
|
|
|
|
#ifdef XWFEATURE_RELAY
|
|
|
|
, XP_Bool breakExisting
|
|
|
|
#endif
|
|
|
|
)
|
2007-11-26 02:58:25 +00:00
|
|
|
{
|
2014-10-15 07:26:18 -07:00
|
|
|
CommsConnType typ;
|
2022-09-03 19:14:40 -07:00
|
|
|
for ( XP_U32 st = 0; addr_iter( &comms->selfAddr, &typ, &st ); ) {
|
2014-10-15 07:26:18 -07:00
|
|
|
switch( typ ) {
|
2006-10-10 01:34:37 +00:00
|
|
|
#ifdef XWFEATURE_RELAY
|
2014-10-15 07:26:18 -07:00
|
|
|
case COMMS_CONN_RELAY:
|
|
|
|
if ( breakExisting
|
|
|
|
|| COMMS_RELAYSTATE_UNCONNECTED == comms->rr.relayState ) {
|
2020-04-25 17:47:07 -07:00
|
|
|
set_relay_state( comms, xwe, COMMS_RELAYSTATE_UNCONNECTED );
|
|
|
|
if ( !relayConnect( comms, xwe ) ) {
|
2022-04-29 09:19:44 -07:00
|
|
|
XP_LOGFF( "relayConnect failed" );
|
2020-04-26 11:28:18 -07:00
|
|
|
set_reset_timer( comms, xwe );
|
2014-10-15 07:26:18 -07:00
|
|
|
}
|
2009-08-30 15:25:13 +00:00
|
|
|
}
|
2014-10-15 07:26:18 -07:00
|
|
|
break;
|
2006-10-10 01:34:37 +00:00
|
|
|
#endif
|
2014-10-15 07:26:18 -07:00
|
|
|
#if defined XWFEATURE_IP_DIRECT || defined XWFEATURE_DIRECTIP
|
|
|
|
case COMMS_CONN_BT:
|
|
|
|
case COMMS_CONN_IP_DIRECT:
|
|
|
|
/* This will only work on host side when there's a single guest! */
|
2020-04-25 17:47:07 -07:00
|
|
|
(void)send_via_bt_or_ip( comms, xwe, BTIPMSG_RESET, CHANNEL_NONE, typ, NULL, 0, NULL );
|
|
|
|
(void)comms_resendAll( comms, xwe, COMMS_CONN_NONE, XP_FALSE );
|
2014-10-15 07:26:18 -07:00
|
|
|
break;
|
2018-07-17 07:05:33 -07:00
|
|
|
#endif
|
|
|
|
#if defined XWFEATURE_SMS
|
|
|
|
case COMMS_CONN_SMS:
|
2020-04-25 17:47:07 -07:00
|
|
|
(void)comms_resendAll( comms, xwe, COMMS_CONN_NONE, XP_FALSE );
|
2018-07-17 07:05:33 -07:00
|
|
|
break;
|
2006-08-26 21:12:10 +00:00
|
|
|
#endif
|
2014-10-15 07:26:18 -07:00
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
2005-04-03 17:00:59 +00:00
|
|
|
}
|
2007-11-18 23:43:27 +00:00
|
|
|
|
2007-11-26 02:58:25 +00:00
|
|
|
setHeartbeatTimer( comms );
|
2020-02-24 21:10:49 +01:00
|
|
|
} /* sendConnect */
|
2005-04-03 17:00:59 +00:00
|
|
|
|
2005-03-19 22:01:38 +00:00
|
|
|
static void
|
2022-09-03 18:47:04 -07:00
|
|
|
addrToStreamOne( XWStreamCtxt* stream, CommsConnType typ,
|
|
|
|
const CommsAddrRec* addrP )
|
2005-03-19 22:01:38 +00:00
|
|
|
{
|
2014-09-25 20:05:37 -07:00
|
|
|
switch( typ ) {
|
2007-12-07 07:50:19 +00:00
|
|
|
case COMMS_CONN_NONE:
|
2007-12-09 01:59:15 +00:00
|
|
|
/* nothing to write */
|
2005-03-19 22:01:38 +00:00
|
|
|
break;
|
|
|
|
case COMMS_CONN_BT:
|
2014-09-25 20:05:37 -07:00
|
|
|
stringToStream( stream, addrP->u.bt.hostName );
|
2006-09-23 16:04:53 +00:00
|
|
|
/* sizeof(.bits) below defeats ARM's padding. */
|
2014-09-25 20:05:37 -07:00
|
|
|
stringToStream( stream, addrP->u.bt.btAddr.chars );
|
2006-09-08 07:14:24 +00:00
|
|
|
break;
|
2005-03-19 22:01:38 +00:00
|
|
|
case COMMS_CONN_IR:
|
|
|
|
/* nothing to save */
|
|
|
|
break;
|
2007-11-26 02:58:25 +00:00
|
|
|
case COMMS_CONN_IP_DIRECT:
|
2014-09-25 20:05:37 -07:00
|
|
|
stringToStream( stream, addrP->u.ip.hostName_ip );
|
|
|
|
stream_putU32( stream, addrP->u.ip.ipAddr_ip );
|
|
|
|
stream_putU16( stream, addrP->u.ip.port_ip );
|
2005-03-19 22:01:38 +00:00
|
|
|
break;
|
|
|
|
case COMMS_CONN_RELAY:
|
2022-10-19 20:33:30 -07:00
|
|
|
if ( stream_getVersion( stream ) < STREAM_VERS_NORELAY ) {
|
|
|
|
IpRelay ip_relay = {{0}};
|
2022-09-28 07:24:45 -07:00
|
|
|
#ifdef XWFEATURE_RELAY
|
2022-10-19 20:33:30 -07:00
|
|
|
ip_relay = addrP->u.ip_relay;
|
2022-09-08 09:57:11 -07:00
|
|
|
#endif
|
2022-10-19 20:33:30 -07:00
|
|
|
stringToStream( stream, ip_relay.invite );
|
|
|
|
stringToStream( stream, ip_relay.hostName );
|
|
|
|
stream_putU32( stream, ip_relay.ipAddr );
|
|
|
|
stream_putU16( stream, ip_relay.port );
|
|
|
|
stream_putBits( stream, 1, ip_relay.seeksPublicRoom );
|
|
|
|
stream_putBits( stream, 1, ip_relay.advertiseRoom );
|
|
|
|
}
|
2022-09-28 07:24:45 -07:00
|
|
|
break;
|
2008-12-29 01:35:29 +00:00
|
|
|
case COMMS_CONN_SMS:
|
2014-09-25 20:05:37 -07:00
|
|
|
stringToStream( stream, addrP->u.sms.phone );
|
|
|
|
stream_putU16( stream, addrP->u.sms.port );
|
2008-12-29 01:35:29 +00:00
|
|
|
break;
|
2016-11-14 08:06:53 -08:00
|
|
|
case COMMS_CONN_P2P:
|
|
|
|
stringToStream( stream, addrP->u.p2p.mac_addr );
|
|
|
|
break;
|
2019-11-23 10:53:51 -08:00
|
|
|
case COMMS_CONN_NFC:
|
|
|
|
break;
|
2020-05-20 13:58:53 -07:00
|
|
|
case COMMS_CONN_MQTT:
|
2022-09-03 18:47:04 -07:00
|
|
|
XP_ASSERT( 0 != addrP->u.mqtt.devID );
|
2020-05-20 13:58:53 -07:00
|
|
|
stream_putBytes( stream, &addrP->u.mqtt.devID, sizeof(addrP->u.mqtt.devID) );
|
|
|
|
break;
|
2009-02-28 19:27:45 +00:00
|
|
|
default:
|
2022-09-28 07:24:45 -07:00
|
|
|
XP_LOGFF( "unexpected typ: %s", ConnType2Str(typ) );
|
2023-01-04 14:13:18 -08:00
|
|
|
XP_ASSERT(0);
|
2009-02-28 19:27:45 +00:00
|
|
|
break;
|
2005-03-19 22:01:38 +00:00
|
|
|
}
|
2014-09-25 20:05:37 -07:00
|
|
|
} /* addrToStreamOne */
|
|
|
|
|
2014-11-02 17:24:37 -08:00
|
|
|
void
|
2014-09-25 20:05:37 -07:00
|
|
|
addrToStream( XWStreamCtxt* stream, const CommsAddrRec* addrP )
|
|
|
|
{
|
2023-02-16 21:36:46 -08:00
|
|
|
XP_ASSERT( 0 < stream_getVersion(stream) ); /* must be set already */
|
2022-10-19 20:33:30 -07:00
|
|
|
XP_U16 conTypes = addrP->_conTypes;
|
|
|
|
types_rmType( &conTypes, COMMS_CONN_RELAY );
|
|
|
|
stream_putU8( stream, conTypes );
|
2014-09-25 20:05:37 -07:00
|
|
|
|
|
|
|
CommsConnType typ;
|
2022-10-19 20:33:30 -07:00
|
|
|
for ( XP_U32 st = 0; types_iter( conTypes, &typ, &st ); ) {
|
2014-09-25 20:05:37 -07:00
|
|
|
addrToStreamOne( stream, typ, addrP );
|
|
|
|
}
|
|
|
|
}
|
2005-03-19 22:01:38 +00:00
|
|
|
|
2023-02-01 19:08:04 -08:00
|
|
|
typedef struct _E2SData {
|
|
|
|
CommsCtxt* comms;
|
|
|
|
XWStreamCtxt* stream;
|
2023-10-26 12:25:59 -07:00
|
|
|
XP_U16 queueLen;
|
2023-02-01 19:08:04 -08:00
|
|
|
} E2SData;
|
|
|
|
|
|
|
|
static ForEachAct
|
|
|
|
elemToStream( MsgQueueElem* elem, void* closure )
|
|
|
|
{
|
2023-10-26 12:25:59 -07:00
|
|
|
if ( 0 == elem->msgID && 0 < elem->smp.len ) {
|
|
|
|
/* Maybe don't wrte non-invites with msgID of 0 */
|
|
|
|
XP_LOGFF( "skipping apparent ACK elem" );
|
2023-02-01 19:08:04 -08:00
|
|
|
} else {
|
2023-10-26 12:25:59 -07:00
|
|
|
E2SData* e2sp = (E2SData*)closure;
|
|
|
|
CommsCtxt* comms = e2sp->comms;
|
|
|
|
|
|
|
|
++e2sp->queueLen;
|
|
|
|
|
|
|
|
XWStreamCtxt* stream = e2sp->stream;
|
|
|
|
writeChannelNo( stream, elem->channelNo );
|
|
|
|
stream_putU32VL( stream, elem->msgID );
|
|
|
|
|
|
|
|
stream_putU32VL( stream, elem->smp.len );
|
|
|
|
stream_putU32( stream, elem->smp.createdStamp );
|
|
|
|
if ( 0 == elem->smp.len ) {
|
|
|
|
XP_ASSERT( 0 == elem->msgID );
|
|
|
|
XWStreamCtxt* nliStream = mem_stream_make_raw( MPPARM(comms->mpool)
|
|
|
|
dutil_getVTManager(comms->dutil));
|
|
|
|
NetLaunchInfo nli;
|
|
|
|
XP_MEMCPY( &nli, elem->smp.buf, sizeof(nli) );
|
|
|
|
nli_saveToStream( &nli, nliStream );
|
|
|
|
XP_U16 nliLen = stream_getSize( nliStream );
|
|
|
|
stream_putU32VL( stream, nliLen );
|
|
|
|
stream_getFromStream( stream, nliStream, nliLen );
|
|
|
|
stream_destroy( nliStream );
|
|
|
|
} else {
|
|
|
|
stream_putBytes( stream, elem->smp.buf, elem->smp.len );
|
|
|
|
// XP_LOGFF( "wrote non-invite with id: %d", elem->msgID );
|
|
|
|
}
|
2023-02-01 19:08:04 -08:00
|
|
|
}
|
|
|
|
return FEA_OK;
|
|
|
|
}
|
|
|
|
|
2003-11-01 05:35:29 +00:00
|
|
|
void
|
2023-02-03 22:00:47 -08:00
|
|
|
comms_writeToStream( CommsCtxt* comms, XWStreamCtxt* stream, XP_U16 saveToken )
|
2003-11-01 05:35:29 +00:00
|
|
|
{
|
2005-03-08 05:33:38 +00:00
|
|
|
XP_U16 nAddrRecs;
|
2003-11-01 05:35:29 +00:00
|
|
|
AddressRecord* rec;
|
|
|
|
|
2022-10-07 09:23:09 -07:00
|
|
|
listRecs( comms, __func__ );
|
|
|
|
|
2015-07-06 20:28:16 -07:00
|
|
|
stream_setVersion( stream, CUR_STREAM_VERS );
|
|
|
|
|
2020-09-26 10:12:29 -07:00
|
|
|
stream_putU8( stream, comms->flags );
|
2023-02-16 21:36:46 -08:00
|
|
|
logAddrComms( comms, &comms->selfAddr, __func__ );
|
2022-09-03 19:14:40 -07:00
|
|
|
addrToStream( stream, &comms->selfAddr );
|
2015-07-06 20:28:16 -07:00
|
|
|
stream_putBits( stream, 4, comms->rr.nPlayersHere );
|
|
|
|
stream_putBits( stream, 4, comms->rr.nPlayersTotal );
|
2005-03-19 22:01:38 +00:00
|
|
|
|
2003-11-01 05:35:29 +00:00
|
|
|
stream_putU32( stream, comms->connID );
|
2023-02-05 08:59:37 -08:00
|
|
|
stream_putU16( stream, comms->streamVersion );
|
2022-10-08 12:26:00 -07:00
|
|
|
writeChannelNo( stream, comms->nextChannelNo );
|
2019-01-08 08:16:50 -08:00
|
|
|
XP_U16 channelSeed = comms_getChannelSeed( comms ); /* force creation */
|
|
|
|
stream_putU16( stream, channelSeed );
|
2012-11-13 21:25:03 -08:00
|
|
|
stream_putU16( stream, comms->resendBackoff );
|
|
|
|
stream_putU32( stream, comms->nextResend );
|
2022-09-03 19:14:40 -07:00
|
|
|
if ( addr_hasType( &comms->selfAddr, COMMS_CONN_RELAY ) ) {
|
2014-02-07 21:12:45 -08:00
|
|
|
stream_putU8( stream, comms->rr.myHostID );
|
2020-08-24 16:16:20 -07:00
|
|
|
XP_LOGFF( "stored myHostID: %d", comms->rr.myHostID );
|
2014-02-07 21:12:45 -08:00
|
|
|
stringToStream( stream, comms->rr.connName );
|
2006-10-05 01:17:03 +00:00
|
|
|
}
|
2005-03-19 22:01:38 +00:00
|
|
|
|
2023-10-26 12:25:59 -07:00
|
|
|
/* Next field is queueLen, but we don't know that until after we write the
|
|
|
|
queue, since ACKs are not persisted. */
|
|
|
|
XWStreamCtxt* tmpStream = mem_stream_make_raw( MPPARM(comms->mpool)
|
|
|
|
dutil_getVTManager(comms->dutil));
|
|
|
|
stream_setVersion( tmpStream, CUR_STREAM_VERS );
|
2003-11-01 05:35:29 +00:00
|
|
|
|
|
|
|
nAddrRecs = countAddrRecs(comms);
|
2023-10-26 12:25:59 -07:00
|
|
|
stream_putU8( tmpStream, (XP_U8)nAddrRecs );
|
2003-11-01 05:35:29 +00:00
|
|
|
|
2019-11-01 07:22:56 -07:00
|
|
|
#ifdef LOG_COMMS_MSGNOS
|
|
|
|
int ii = 0;
|
|
|
|
#endif
|
2003-11-01 05:35:29 +00:00
|
|
|
for ( rec = comms->recs; !!rec; rec = rec->next ) {
|
2005-03-19 22:01:38 +00:00
|
|
|
|
2022-09-03 18:47:04 -07:00
|
|
|
const CommsAddrRec* addr = &rec->addr;
|
2023-02-16 21:36:46 -08:00
|
|
|
logAddrComms( comms, addr, __func__ );
|
2023-10-26 12:25:59 -07:00
|
|
|
addrToStream( tmpStream, addr );
|
2004-10-02 03:47:24 +00:00
|
|
|
|
2023-10-26 12:25:59 -07:00
|
|
|
stream_putU32VL( tmpStream, rec->nextMsgID );
|
|
|
|
stream_putU32VL( tmpStream, rec->lastMsgRcd );
|
|
|
|
stream_putU16( tmpStream, rec->flags );
|
2019-11-01 07:22:56 -07:00
|
|
|
#ifdef LOG_COMMS_MSGNOS
|
2021-12-21 16:23:24 -08:00
|
|
|
XP_LOGFF( "wrote lastMsgRcd of %d for addr %d", rec->lastMsgRcd, ii++ );
|
2019-11-01 07:22:56 -07:00
|
|
|
#endif
|
2023-10-26 12:25:59 -07:00
|
|
|
stream_putU16( tmpStream, (XP_U16)rec->lastMsgAckd );
|
|
|
|
writeChannelNo( tmpStream, rec->channelNo );
|
2022-09-03 18:47:04 -07:00
|
|
|
if ( addr_hasType( addr, COMMS_CONN_RELAY ) ) {
|
2022-10-01 08:43:10 -07:00
|
|
|
XP_ASSERT(0);
|
2023-10-26 12:25:59 -07:00
|
|
|
stream_putU8( tmpStream, rec->rr.hostID ); /* unneeded unless RELAY */
|
2006-10-05 01:17:03 +00:00
|
|
|
}
|
2003-11-01 05:35:29 +00:00
|
|
|
}
|
|
|
|
|
2023-10-26 12:25:59 -07:00
|
|
|
E2SData e2sd = { .comms = comms, .stream = tmpStream, };
|
2023-02-01 19:08:04 -08:00
|
|
|
forEachElem( comms, elemToStream, &e2sd );
|
2003-11-01 05:35:29 +00:00
|
|
|
|
2023-10-26 12:25:59 -07:00
|
|
|
XP_ASSERT( e2sd.queueLen <= 255 );
|
|
|
|
XP_ASSERT( e2sd.queueLen <= comms->queueLen );
|
|
|
|
stream_putU8( stream, (XP_U8)e2sd.queueLen );
|
|
|
|
stream_getFromStream( stream, tmpStream, stream_getSize(tmpStream) );
|
|
|
|
stream_destroy( tmpStream );
|
|
|
|
|
2021-12-22 13:01:22 -08:00
|
|
|
/* This writes 2 bytes instead of 1 if it were smarter. Not worth the work
|
|
|
|
* to fix. */
|
2019-11-30 18:39:11 -08:00
|
|
|
for ( CommsConnType typ = (CommsConnType)0; typ < VSIZE(comms->disableds); ++typ ) {
|
2022-09-03 19:14:40 -07:00
|
|
|
if ( typ < COMMS_CONN_NFC || addr_hasType( &comms->selfAddr, typ ) ) {
|
2019-11-30 18:39:11 -08:00
|
|
|
for ( int ii = 0; ii < VSIZE(comms->disableds[0]); ++ii ) {
|
|
|
|
stream_putBits( stream, 1, comms->disableds[typ][ii] ? 1 : 0 );
|
|
|
|
}
|
2017-05-26 07:30:42 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-09-10 07:31:45 -07:00
|
|
|
comms->lastSaveToken = saveToken;
|
2003-11-01 05:35:29 +00:00
|
|
|
} /* comms_writeToStream */
|
|
|
|
|
2012-11-13 21:25:03 -08:00
|
|
|
static void
|
|
|
|
resetBackoff( CommsCtxt* comms )
|
|
|
|
{
|
2020-04-06 16:34:02 -07:00
|
|
|
XP_LOGFF( "resetting backoff" );
|
2012-11-13 21:25:03 -08:00
|
|
|
comms->resendBackoff = 0;
|
|
|
|
comms->nextResend = 0;
|
|
|
|
}
|
|
|
|
|
2012-09-10 07:31:45 -07:00
|
|
|
void
|
2020-04-25 17:47:07 -07:00
|
|
|
comms_saveSucceeded( CommsCtxt* comms, XWEnv xwe, XP_U16 saveToken )
|
2012-09-10 07:31:45 -07:00
|
|
|
{
|
2023-01-06 07:52:41 -08:00
|
|
|
THREAD_CHECK_START(comms);
|
2022-04-29 09:19:44 -07:00
|
|
|
XP_LOGFF( "(saveToken=%d)", saveToken );
|
2012-09-10 07:31:45 -07:00
|
|
|
XP_ASSERT( !!comms );
|
|
|
|
if ( saveToken == comms->lastSaveToken ) {
|
|
|
|
AddressRecord* rec;
|
|
|
|
for ( rec = comms->recs; !!rec; rec = rec->next ) {
|
2022-04-29 09:19:44 -07:00
|
|
|
XP_LOGFF( "lastSave matches; updating lastMsgSaved (%d) to "
|
|
|
|
"lastMsgRcd (%d)", rec->lastMsgSaved, rec->lastMsgRcd );
|
2012-09-10 07:31:45 -07:00
|
|
|
rec->lastMsgSaved = rec->lastMsgRcd;
|
|
|
|
}
|
|
|
|
#ifdef XWFEATURE_COMMSACK
|
2020-04-25 17:47:07 -07:00
|
|
|
comms_ackAny( comms, xwe ); /* might not want this for all transports */
|
2012-09-10 07:31:45 -07:00
|
|
|
#endif
|
|
|
|
}
|
2023-01-06 07:52:41 -08:00
|
|
|
THREAD_CHECK_END();
|
2012-09-10 07:31:45 -07:00
|
|
|
}
|
|
|
|
|
2005-06-27 05:45:28 +00:00
|
|
|
void
|
2022-09-03 19:14:40 -07:00
|
|
|
comms_getSelfAddr( const CommsCtxt* comms, CommsAddrRec* addr )
|
2003-11-01 05:35:29 +00:00
|
|
|
{
|
2005-04-03 02:49:04 +00:00
|
|
|
XP_ASSERT( !!comms );
|
2022-09-03 19:14:40 -07:00
|
|
|
XP_MEMCPY( addr, &comms->selfAddr, sizeof(*addr) );
|
2003-11-01 05:35:29 +00:00
|
|
|
} /* comms_getAddr */
|
2022-09-06 18:21:53 -07:00
|
|
|
|
|
|
|
XP_Bool
|
|
|
|
comms_getHostAddr( const CommsCtxt* comms, CommsAddrRec* addr )
|
|
|
|
{
|
|
|
|
XP_ASSERT( !!comms );
|
|
|
|
XP_Bool haveAddr = !comms->isServer
|
|
|
|
&& !!comms->recs
|
|
|
|
&& !comms->recs->next
|
|
|
|
;
|
|
|
|
if ( haveAddr ) {
|
|
|
|
XP_MEMCPY( addr, &comms->recs->addr, sizeof(*addr) );
|
|
|
|
}
|
|
|
|
return haveAddr;
|
|
|
|
} /* comms_getAddr */
|
2003-11-01 05:35:29 +00:00
|
|
|
|
2020-09-05 14:56:03 -07:00
|
|
|
void
|
|
|
|
comms_addMQTTDevID( CommsCtxt* comms, XP_PlayerAddr channelNo,
|
|
|
|
const MQTTDevID* devID )
|
2020-08-22 16:02:31 -07:00
|
|
|
{
|
2023-01-06 07:52:41 -08:00
|
|
|
THREAD_CHECK_START(comms);
|
2020-10-21 21:10:09 -07:00
|
|
|
#ifdef NO_ADD_MQTT_TO_ALL /* set for (usually) BT testing on Android */
|
|
|
|
XP_LOGFF("ifdef'd out");
|
|
|
|
XP_USE( comms );
|
|
|
|
XP_USE( channelNo );
|
|
|
|
XP_USE( devID );
|
|
|
|
#else
|
2022-10-07 15:04:24 -07:00
|
|
|
CNO_FMT( cbuf, channelNo );
|
|
|
|
XP_LOGFF( "(channelNo: %s, devID: " MQTTDevID_FMT ")", cbuf, *devID );
|
2020-08-22 16:02:31 -07:00
|
|
|
XP_Bool found = XP_FALSE;
|
|
|
|
for ( AddressRecord* rec = comms->recs; !!rec && !found; rec = rec->next ) {
|
2022-02-12 13:27:24 -08:00
|
|
|
found = (rec->channelNo & ~CHANNEL_MASK) == (channelNo & ~CHANNEL_MASK);
|
2020-08-22 16:02:31 -07:00
|
|
|
if ( found ) {
|
2023-01-15 16:18:26 -08:00
|
|
|
if ( !addr_hasType( &comms->selfAddr, COMMS_CONN_MQTT ) ) {
|
|
|
|
XP_LOGFF( "not adding mqtt because game doesn't allow it" );
|
|
|
|
} else if ( addr_hasType( &rec->addr, COMMS_CONN_MQTT ) ) {
|
2020-08-22 16:02:31 -07:00
|
|
|
XP_ASSERT( *devID == rec->addr.u.mqtt.devID );
|
2022-09-03 18:47:04 -07:00
|
|
|
} else {
|
|
|
|
CommsAddrRec tmp = {0};
|
|
|
|
addr_setType( &tmp, COMMS_CONN_MQTT );
|
|
|
|
tmp.u.mqtt.devID = *devID;
|
|
|
|
ASSERT_ADDR_OK( &tmp );
|
|
|
|
|
|
|
|
augmentAddrIntrnl( comms, &rec->addr, &tmp, XP_TRUE );
|
|
|
|
ASSERT_ADDR_OK( &rec->addr );
|
2022-10-07 15:04:24 -07:00
|
|
|
CNO_FMT( cbuf, channelNo );
|
|
|
|
XP_LOGFF( "added for channel %s", cbuf );
|
2020-08-22 16:02:31 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if ( !found ) {
|
|
|
|
XP_LOGFF( "unable to augment address!!" );
|
2022-09-03 18:47:04 -07:00
|
|
|
XP_ASSERT(0);
|
2020-08-22 16:02:31 -07:00
|
|
|
}
|
2020-10-21 21:10:09 -07:00
|
|
|
#endif
|
2023-01-06 07:52:41 -08:00
|
|
|
THREAD_CHECK_END();
|
2020-08-22 16:02:31 -07:00
|
|
|
}
|
|
|
|
|
2012-04-18 23:23:36 -07:00
|
|
|
void
|
2023-02-16 21:36:46 -08:00
|
|
|
comms_getAddrs( const CommsCtxt* comms, CommsAddrRec addrs[],
|
|
|
|
XP_U16* nAddrs )
|
2012-04-18 23:23:36 -07:00
|
|
|
{
|
2023-02-16 21:36:46 -08:00
|
|
|
XP_U16 count = 0;
|
|
|
|
for ( AddressRecord* recs = comms->recs; !!recs; recs = recs->next ) {
|
|
|
|
if ( count < *nAddrs ) {
|
|
|
|
XP_MEMCPY( &addrs[count], &recs->addr, sizeof(addrs[count]) );
|
|
|
|
logAddrComms( comms, &addrs[count], __func__ );
|
|
|
|
}
|
|
|
|
++count;
|
|
|
|
}
|
|
|
|
*nAddrs = count;
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
comms_getChannelAddr( const CommsCtxt* comms, XP_PlayerAddr channelNo,
|
|
|
|
CommsAddrRec* addr )
|
|
|
|
{
|
|
|
|
XP_U16 masked = channelNo & CHANNEL_MASK;
|
|
|
|
XP_Bool found = XP_FALSE;
|
|
|
|
for ( const AddressRecord* rec = comms->recs;
|
|
|
|
!found && !!rec; rec = rec->next ) {
|
|
|
|
found = (rec->channelNo & CHANNEL_MASK) == masked;
|
|
|
|
if ( found ) {
|
|
|
|
XP_LOGFF( "writing addr for channel %X", channelNo );
|
|
|
|
*addr = rec->addr;
|
|
|
|
logAddrComms( comms, addr, __func__ );
|
|
|
|
}
|
2012-04-18 23:23:36 -07:00
|
|
|
}
|
2023-02-16 21:36:46 -08:00
|
|
|
XP_ASSERT( found );
|
2012-04-18 23:23:36 -07:00
|
|
|
}
|
|
|
|
|
2023-12-18 17:08:51 -08:00
|
|
|
static XP_Bool
|
|
|
|
addrs_same( const CommsAddrRec* addr1, const CommsAddrRec* addr2 )
|
|
|
|
{
|
|
|
|
/* Empty addresses are the same only if both are empty */
|
|
|
|
XP_Bool same = addr1->_conTypes == 0 && addr2->_conTypes == 0;
|
|
|
|
|
|
|
|
CommsConnType typ;
|
|
|
|
for ( XP_U32 st = 0; !same && addr_iter( addr1, &typ, &st ); ) {
|
|
|
|
if ( addr_hasType( addr2, typ ) ) {
|
|
|
|
switch ( typ ) {
|
|
|
|
case COMMS_CONN_MQTT:
|
|
|
|
same = addr1->u.mqtt.devID == addr2->u.mqtt.devID;
|
|
|
|
break;
|
|
|
|
case COMMS_CONN_SMS:
|
|
|
|
same = addr1->u.sms.port == addr2->u.sms.port
|
|
|
|
&& 0 == XP_STRCMP(addr1->u.sms.phone, addr2->u.sms.phone );
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
XP_LOGFF( "ignoring %s", ConnType2Str(typ) );
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return same;
|
|
|
|
}
|
|
|
|
|
|
|
|
XP_Bool
|
|
|
|
comms_addrsAreSame( const CommsCtxt* XP_UNUSED(comms),
|
|
|
|
const CommsAddrRec* addr1,
|
|
|
|
const CommsAddrRec* addr2 )
|
|
|
|
{
|
|
|
|
XP_Bool result = addrs_same( addr1, addr2 );
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
2023-11-17 19:39:45 -08:00
|
|
|
typedef struct _NonAcks {
|
|
|
|
int count;
|
|
|
|
} NonAcks;
|
|
|
|
|
|
|
|
static ForEachAct
|
|
|
|
countNonAcks( MsgQueueElem* elem, void* closure )
|
|
|
|
{
|
|
|
|
if ( IS_INVITE(elem) || 0 != elem->msgID ) {
|
|
|
|
NonAcks* nap = (NonAcks*)closure;
|
|
|
|
++nap->count;
|
|
|
|
}
|
|
|
|
return FEA_OK;
|
|
|
|
}
|
|
|
|
|
2014-02-28 18:59:12 -08:00
|
|
|
XP_U16
|
2023-11-06 12:51:10 -08:00
|
|
|
comms_countPendingPackets( const CommsCtxt* comms, XP_Bool* quashed )
|
2014-02-28 18:59:12 -08:00
|
|
|
{
|
2023-11-06 12:51:10 -08:00
|
|
|
if ( !!quashed ) {
|
|
|
|
*quashed = QUASHED(comms);
|
|
|
|
}
|
2023-11-17 19:39:45 -08:00
|
|
|
|
|
|
|
NonAcks na = {0};
|
|
|
|
forEachElem( (CommsCtxt*)comms, countNonAcks, &na );
|
|
|
|
|
|
|
|
// XP_LOGFF( "=> %d (queueLen = %d)", na.count, comms->queueLen );
|
|
|
|
return na.count;
|
2014-02-28 18:59:12 -08:00
|
|
|
}
|
|
|
|
|
2011-08-10 18:28:34 -07:00
|
|
|
static XP_Bool
|
|
|
|
formatRelayID( const CommsCtxt* comms, XWHostID hostID,
|
|
|
|
XP_UCHAR* buf, XP_U16* lenp )
|
|
|
|
{
|
|
|
|
XP_U16 strln = 1 + XP_SNPRINTF( buf, *lenp, "%s/%d",
|
2014-02-07 21:12:45 -08:00
|
|
|
comms->rr.connName, hostID );
|
2011-08-10 18:28:34 -07:00
|
|
|
XP_ASSERT( *lenp >= strln );
|
|
|
|
*lenp = strln;
|
|
|
|
return XP_TRUE;
|
|
|
|
}
|
|
|
|
|
2022-03-14 07:42:47 +01:00
|
|
|
#ifdef XWFEATURE_RELAY
|
|
|
|
static XP_Bool
|
|
|
|
haveRelayID( const CommsCtxt* comms )
|
|
|
|
{
|
|
|
|
XP_Bool result = 0 != comms->rr.connName[0]
|
|
|
|
&& comms->rr.myHostID != HOST_ID_NONE;
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
2015-07-06 20:28:16 -07:00
|
|
|
XP_Bool
|
|
|
|
comms_formatRelayID( const CommsCtxt* comms, XP_U16 indx,
|
|
|
|
XP_UCHAR* buf, XP_U16* lenp )
|
|
|
|
{
|
2022-04-29 09:19:44 -07:00
|
|
|
XP_LOGFF( "(indx=%d)", indx );
|
2015-07-06 20:28:16 -07:00
|
|
|
XWHostID hostID = HOST_ID_SERVER;
|
|
|
|
if ( comms->isServer ) {
|
|
|
|
hostID += 1 + indx;
|
|
|
|
}
|
|
|
|
XP_Bool success = formatRelayID( comms, hostID, buf, lenp );
|
2022-04-29 09:19:44 -07:00
|
|
|
XP_LOGFF( "(%d) => %s", indx, buf );
|
2015-07-06 20:28:16 -07:00
|
|
|
return success;
|
|
|
|
}
|
|
|
|
|
2011-08-10 18:28:34 -07:00
|
|
|
/* Get *my* "relayID", a combo of connname and host id */
|
2010-08-22 12:16:57 -07:00
|
|
|
XP_Bool
|
2010-10-11 06:20:30 -07:00
|
|
|
comms_getRelayID( const CommsCtxt* comms, XP_UCHAR* buf, XP_U16* lenp )
|
2010-08-22 12:16:57 -07:00
|
|
|
{
|
2011-08-10 18:28:34 -07:00
|
|
|
XP_Bool result = haveRelayID( comms )
|
2014-02-07 21:12:45 -08:00
|
|
|
&& formatRelayID( comms, comms->rr.myHostID, buf, lenp );
|
2011-08-10 18:28:34 -07:00
|
|
|
return result;
|
2010-08-22 12:16:57 -07:00
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2021-03-19 18:51:34 -07:00
|
|
|
static void
|
|
|
|
formatMsgNo( const CommsCtxt* comms, const MsgQueueElem* elem,
|
|
|
|
XP_UCHAR* buf, XP_U16 len )
|
|
|
|
{
|
|
|
|
XP_SNPRINTF( buf, len, "%d:%d", comms->rr.myHostID, elem->msgID );
|
|
|
|
}
|
|
|
|
|
2014-10-15 07:26:18 -07:00
|
|
|
CommsConnTypes
|
|
|
|
comms_getConTypes( const CommsCtxt* comms )
|
2003-11-01 05:35:29 +00:00
|
|
|
{
|
2012-03-06 06:23:04 -08:00
|
|
|
CommsConnType typ;
|
|
|
|
if ( !!comms ) {
|
2022-09-03 19:14:40 -07:00
|
|
|
typ = comms->selfAddr._conTypes;
|
2012-03-06 06:23:04 -08:00
|
|
|
} else {
|
|
|
|
typ = COMMS_CONN_NONE;
|
2022-04-29 09:19:44 -07:00
|
|
|
XP_LOGFF( "returning COMMS_CONN_NONE for null comms" );
|
2012-02-14 16:54:19 -08:00
|
|
|
}
|
|
|
|
return typ;
|
2015-07-06 20:28:16 -07:00
|
|
|
} /* comms_getConTypes */
|
2003-11-01 05:35:29 +00:00
|
|
|
|
2020-03-14 18:52:28 -07:00
|
|
|
void
|
|
|
|
comms_dropHostAddr( CommsCtxt* comms, CommsConnType typ )
|
|
|
|
{
|
2023-01-06 07:52:41 -08:00
|
|
|
THREAD_CHECK_START(comms);
|
2022-09-03 19:14:40 -07:00
|
|
|
addr_rmType( &comms->selfAddr, typ );
|
|
|
|
ASSERT_ADDR_OK( &comms->selfAddr );
|
2023-01-06 07:52:41 -08:00
|
|
|
THREAD_CHECK_END();
|
2020-03-14 18:52:28 -07:00
|
|
|
}
|
|
|
|
|
2006-09-08 07:14:24 +00:00
|
|
|
XP_Bool
|
2023-12-05 08:43:22 -08:00
|
|
|
comms_getIsHost( const CommsCtxt* comms )
|
2006-09-08 07:14:24 +00:00
|
|
|
{
|
2007-09-15 13:46:59 +00:00
|
|
|
XP_ASSERT( !!comms );
|
2006-09-08 07:14:24 +00:00
|
|
|
return comms->isServer;
|
|
|
|
}
|
|
|
|
|
2022-09-06 12:16:29 -07:00
|
|
|
static MsgQueueElem*
|
|
|
|
makeNewElem( const CommsCtxt* comms, XWEnv xwe, MsgID msgID,
|
|
|
|
XP_PlayerAddr channelNo )
|
|
|
|
{
|
|
|
|
MsgQueueElem* newElem = (MsgQueueElem*)XP_CALLOC( comms->mpool,
|
|
|
|
sizeof( *newElem ) );
|
2023-03-18 15:15:34 -07:00
|
|
|
newElem->smp.createdStamp = dutil_getCurSeconds( comms->dutil, xwe );
|
2022-09-06 12:16:29 -07:00
|
|
|
newElem->channelNo = channelNo;
|
|
|
|
newElem->msgID = msgID;
|
|
|
|
return newElem;
|
|
|
|
}
|
|
|
|
|
2007-11-26 02:58:25 +00:00
|
|
|
static MsgQueueElem*
|
2023-01-06 07:52:41 -08:00
|
|
|
makeElemWithID( const CommsCtxt* comms, XWEnv xwe, MsgID msgID, AddressRecord* rec,
|
2007-11-26 02:58:25 +00:00
|
|
|
XP_PlayerAddr channelNo, XWStreamCtxt* stream )
|
2003-11-01 05:35:29 +00:00
|
|
|
{
|
2014-11-06 18:44:19 -08:00
|
|
|
CNO_FMT( cbuf, channelNo );
|
2021-12-15 10:03:09 -08:00
|
|
|
XP_LOGFF( TAGFMT(%s), TAGPRMS, cbuf );
|
2007-11-18 23:43:27 +00:00
|
|
|
XP_U16 streamSize = NULL == stream? 0 : stream_getSize( stream );
|
2012-09-10 07:31:45 -07:00
|
|
|
MsgID lastMsgSaved = (!!rec)? rec->lastMsgSaved : 0;
|
2022-09-06 12:16:29 -07:00
|
|
|
MsgQueueElem* newElem = makeNewElem( comms, xwe, msgID, channelNo );
|
2003-11-01 05:35:29 +00:00
|
|
|
|
2021-12-29 07:35:43 -08:00
|
|
|
XP_Bool useSmallHeader = !!rec && (COMMS_VERSION == rec->flags);
|
2021-12-21 16:23:24 -08:00
|
|
|
XWStreamCtxt* hdrStream = mem_stream_make_raw( MPPARM(comms->mpool)
|
|
|
|
dutil_getVTManager(comms->dutil));
|
|
|
|
XP_ASSERT( 0L == comms->connID || comms->connID == comms->util->gameInfo->gameID );
|
|
|
|
if ( !useSmallHeader ) {
|
|
|
|
XP_LOGFF( TAGFMT() "putting connID %x", TAGPRMS, comms->connID );
|
|
|
|
stream_putU32( hdrStream, comms->connID );
|
|
|
|
}
|
2005-03-06 17:56:34 +00:00
|
|
|
|
2013-12-17 21:08:27 -08:00
|
|
|
stream_putU16( hdrStream, channelNo );
|
2021-12-21 16:23:24 -08:00
|
|
|
|
|
|
|
if ( useSmallHeader ) {
|
2021-12-28 18:49:56 -08:00
|
|
|
if ( 0 != msgID ) { /* bit in flags says not there */
|
|
|
|
stream_putU32VL( hdrStream, msgID );
|
|
|
|
}
|
|
|
|
stream_putU32VL( hdrStream, lastMsgSaved );
|
2021-12-21 16:23:24 -08:00
|
|
|
#if 0 && defined DEBUG
|
|
|
|
/* Test receiver's ability to skip unexpected header fields */
|
|
|
|
stream_putU8( hdrStream, 0x01 );
|
|
|
|
stream_putU8( hdrStream, 0x02 );
|
|
|
|
stream_putU8( hdrStream, 0x03 );
|
|
|
|
#endif
|
|
|
|
} else {
|
|
|
|
stream_putU32( hdrStream, msgID );
|
|
|
|
stream_putU32( hdrStream, lastMsgSaved );
|
|
|
|
}
|
2021-12-15 10:03:09 -08:00
|
|
|
XP_LOGFF( TAGFMT() "put lastMsgSaved: %d", TAGPRMS, lastMsgSaved );
|
2012-04-13 20:21:59 -07:00
|
|
|
if ( !!rec ) {
|
2012-09-10 07:31:45 -07:00
|
|
|
rec->lastMsgAckd = lastMsgSaved;
|
2012-04-13 20:21:59 -07:00
|
|
|
}
|
2003-11-01 05:35:29 +00:00
|
|
|
|
2021-12-21 16:23:24 -08:00
|
|
|
/* Now we'll use a third stream to combine them all */
|
|
|
|
XP_U16 headerLen = stream_getSize( hdrStream );
|
2021-12-28 18:49:56 -08:00
|
|
|
XP_U16 flags = makeFlags( comms, headerLen, msgID );
|
2021-12-21 16:23:24 -08:00
|
|
|
XWStreamCtxt* msgStream = mem_stream_make_raw( MPPARM(comms->mpool)
|
|
|
|
dutil_getVTManager(comms->dutil));
|
|
|
|
if ( useSmallHeader ) {
|
|
|
|
XP_ASSERT( HAS_VERSION_FLAG != flags );
|
|
|
|
} else {
|
|
|
|
stream_putU16( msgStream, HAS_VERSION_FLAG );
|
|
|
|
}
|
|
|
|
stream_putU16( msgStream, flags );
|
2003-11-01 05:35:29 +00:00
|
|
|
|
2021-12-21 16:23:24 -08:00
|
|
|
stream_getFromStream( msgStream, hdrStream, stream_getSize(hdrStream) );
|
2023-02-03 22:00:47 -08:00
|
|
|
stream_destroy( hdrStream );
|
2021-12-21 16:23:24 -08:00
|
|
|
|
2007-11-18 23:43:27 +00:00
|
|
|
if ( 0 < streamSize ) {
|
2021-12-21 16:23:24 -08:00
|
|
|
stream_getFromStream( msgStream, stream, streamSize );
|
2007-11-18 23:43:27 +00:00
|
|
|
}
|
2003-11-01 05:35:29 +00:00
|
|
|
|
2023-03-18 15:15:34 -07:00
|
|
|
newElem->smp.len = stream_getSize( msgStream );
|
|
|
|
XP_ASSERT( 0 < newElem->smp.len );
|
|
|
|
newElem->smp.buf = (XP_U8*)XP_MALLOC( comms->mpool, newElem->smp.len );
|
|
|
|
stream_getBytes( msgStream, (XP_U8*)newElem->smp.buf, newElem->smp.len );
|
2023-02-03 22:00:47 -08:00
|
|
|
stream_destroy( msgStream );
|
2021-12-21 16:23:24 -08:00
|
|
|
|
2010-10-18 22:08:04 -07:00
|
|
|
#ifdef COMMS_CHECKSUM
|
2023-03-18 15:15:34 -07:00
|
|
|
newElem->checksum = dutil_md5sum( comms->dutil, xwe, newElem->smp.buf,
|
|
|
|
newElem->smp.len );
|
2010-10-18 20:17:44 -07:00
|
|
|
#endif
|
2023-03-18 15:15:34 -07:00
|
|
|
XP_ASSERT( 0 < newElem->smp.len ); /* else NLI assumptions fail */
|
2022-09-06 12:16:29 -07:00
|
|
|
return newElem;
|
2007-11-26 02:58:25 +00:00
|
|
|
} /* makeElemWithID */
|
2007-11-18 23:43:27 +00:00
|
|
|
|
2022-09-06 12:16:29 -07:00
|
|
|
#ifdef XWFEATURE_COMMS_INVITE
|
|
|
|
static MsgQueueElem*
|
|
|
|
makeInviteElem( CommsCtxt* comms, XWEnv xwe,
|
|
|
|
XP_PlayerAddr channelNo, const NetLaunchInfo* nli )
|
|
|
|
{
|
|
|
|
MsgQueueElem* newElem = makeNewElem( comms, xwe, 0, channelNo );
|
|
|
|
|
2023-03-18 15:15:34 -07:00
|
|
|
XP_ASSERT( 0 == newElem->smp.len ); /* len == 0 signals is NLI */
|
|
|
|
newElem->smp.buf = XP_MALLOC( comms->mpool, sizeof(*nli) );
|
|
|
|
XP_MEMCPY( (XP_U8*)newElem->smp.buf, nli, sizeof(*nli) );
|
2022-09-06 12:16:29 -07:00
|
|
|
# ifdef COMMS_CHECKSUM
|
2023-03-18 15:15:34 -07:00
|
|
|
newElem->checksum = dutil_md5sum( comms->dutil, xwe, newElem->smp.buf,
|
2022-09-06 12:16:29 -07:00
|
|
|
sizeof(*nli) );
|
|
|
|
# endif
|
|
|
|
return newElem;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2010-11-08 17:22:38 -08:00
|
|
|
XP_U16
|
|
|
|
comms_getChannelSeed( CommsCtxt* comms )
|
2009-09-26 14:37:49 +00:00
|
|
|
{
|
2020-01-24 09:05:16 -08:00
|
|
|
XP_U16 result = !!comms ? comms->channelSeed : 0;
|
|
|
|
while ( !!comms && 0 == (result & ~CHANNEL_MASK) ) {
|
|
|
|
result = XP_RANDOM() & ~CHANNEL_MASK;
|
|
|
|
result |= comms->forceChannel;
|
|
|
|
CNO_FMT( cbuf, result );
|
2020-02-24 21:10:49 +01:00
|
|
|
XP_LOGFF( "made seed: %s(%d)", cbuf, result );
|
2020-01-24 09:05:16 -08:00
|
|
|
comms->channelSeed = result;
|
2009-09-26 14:37:49 +00:00
|
|
|
}
|
2020-01-24 09:05:16 -08:00
|
|
|
return result;
|
2009-09-26 14:37:49 +00:00
|
|
|
}
|
|
|
|
|
2022-09-06 12:16:29 -07:00
|
|
|
#ifdef XWFEATURE_COMMS_INVITE
|
|
|
|
/* We're adding invites to comms so they'll be persisted and resent etc. can
|
|
|
|
work in common code. Rule will be there's only one invitation present per
|
|
|
|
channel (remote device.) We'll add a channel if necessary. Then if there is
|
|
|
|
an invitation already there we'll replace it.
|
|
|
|
|
|
|
|
Interesting case is where I send two invitations only one of which can be
|
|
|
|
accepted (as e.g. when I invite one friend, then on not receiving a
|
|
|
|
response decide to invite a different friend instead. In the two-player
|
|
|
|
game case I can detect that there are too many invitations and delete/reuse
|
|
|
|
the previous channel when the new invitation is received. But what about
|
|
|
|
the two-remotes game? Would I delete the oldest that hadn't been responded
|
|
|
|
to? Or maybe keep them all, and when an invitation is responded to, meaning
|
|
|
|
that there's incoming traffic on that channel, I promote it somehow, and
|
|
|
|
when a game is complete (all players present) delete channels that have
|
|
|
|
only outgoing invitations pending.
|
2022-09-08 09:57:11 -07:00
|
|
|
|
|
|
|
Let's use channel 1 for invites. And be prepared to nuke their records once
|
|
|
|
real traffic develops on that channel.
|
2022-09-06 12:16:29 -07:00
|
|
|
*/
|
2022-09-08 09:57:11 -07:00
|
|
|
|
|
|
|
/* Remove any AddressRecord created for invitations, and any MsgQueueElems for it */
|
|
|
|
static void
|
|
|
|
nukeInvites( CommsCtxt* comms, XWEnv xwe, XP_PlayerAddr channelNo )
|
|
|
|
{
|
|
|
|
XP_LOGFF( "(channelNo=0x%X)", channelNo );
|
2023-03-16 07:41:53 -07:00
|
|
|
assertQueueOk( comms );
|
2022-12-27 16:03:08 -08:00
|
|
|
channelNo &= CHANNEL_MASK;
|
2022-09-08 09:57:11 -07:00
|
|
|
|
2022-10-07 09:23:09 -07:00
|
|
|
listRecs( comms, __func__ );
|
|
|
|
|
2022-09-08 09:57:11 -07:00
|
|
|
AddressRecord* deadRec;
|
|
|
|
|
|
|
|
AddressRecord* prevRec = NULL;
|
|
|
|
for ( deadRec = comms->recs; !!deadRec; deadRec = deadRec->next ) {
|
|
|
|
if ( channelNo == deadRec->channelNo ) {
|
|
|
|
// XP_ASSERT( forceChannel == rec->channelNo ); /* should not have high bits */
|
|
|
|
if ( NULL == prevRec ) {
|
|
|
|
comms->recs = deadRec->next;
|
|
|
|
} else {
|
|
|
|
prevRec->next = deadRec->next;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
prevRec = deadRec;
|
|
|
|
}
|
|
|
|
|
|
|
|
if ( !!deadRec ) {
|
2023-03-16 07:41:53 -07:00
|
|
|
XP_ASSERT( !!deadRec->_msgQueueHead ); /* otherwise we'll leak */
|
|
|
|
freeElem( MPPARM(comms->mpool) deadRec->_msgQueueHead );
|
|
|
|
deadRec->_msgQueueHead = NULL;
|
|
|
|
--comms->queueLen;
|
2022-09-08 09:57:11 -07:00
|
|
|
removeFromQueue( comms, xwe, channelNo, 0 );
|
2022-10-07 09:23:09 -07:00
|
|
|
CNO_FMT( cbuf, deadRec->channelNo );
|
|
|
|
XP_LOGFF( "removing rec for %s", cbuf );
|
2023-03-16 07:41:53 -07:00
|
|
|
XP_ASSERT( !deadRec->_msgQueueHead );
|
2022-09-08 09:57:11 -07:00
|
|
|
XP_FREEP( comms->mpool, &deadRec );
|
|
|
|
}
|
2022-10-07 09:23:09 -07:00
|
|
|
|
|
|
|
listRecs( comms, "end of nukeInvites" );
|
2022-09-08 09:57:11 -07:00
|
|
|
} /* nukeInvites */
|
|
|
|
|
2022-10-01 08:43:10 -07:00
|
|
|
static XP_Bool
|
|
|
|
haveRealChannel( const CommsCtxt* comms, XP_PlayerAddr channelNo )
|
|
|
|
{
|
|
|
|
XP_ASSERT( (channelNo & CHANNEL_MASK) == channelNo );
|
|
|
|
XP_Bool found = XP_FALSE;
|
|
|
|
|
|
|
|
for ( AddressRecord* rec = comms->recs; !!rec && !found; rec = rec->next ) {
|
|
|
|
found = (channelNo == (CHANNEL_MASK & rec->channelNo))
|
|
|
|
&& (0 != (rec->channelNo & ~CHANNEL_MASK));
|
|
|
|
}
|
|
|
|
|
|
|
|
CNO_FMT( cbuf, channelNo );
|
|
|
|
XP_LOGFF( "(%s) => %s", cbuf, boolToStr(found) );
|
|
|
|
return found;
|
|
|
|
}
|
|
|
|
|
2022-09-06 12:16:29 -07:00
|
|
|
void
|
|
|
|
comms_invite( CommsCtxt* comms, XWEnv xwe, const NetLaunchInfo* nli,
|
2023-01-30 12:02:46 -08:00
|
|
|
const CommsAddrRec* destAddr, XP_Bool sendNow )
|
2022-09-06 12:16:29 -07:00
|
|
|
{
|
|
|
|
LOG_FUNC();
|
2022-10-20 12:19:33 -07:00
|
|
|
LOGNLI(nli);
|
2022-09-08 09:57:11 -07:00
|
|
|
XP_PlayerAddr forceChannel = nli->forceChannel;
|
2022-12-27 16:03:08 -08:00
|
|
|
XP_ASSERT( 0 < forceChannel && (forceChannel & CHANNEL_MASK) == forceChannel );
|
2022-10-01 08:43:10 -07:00
|
|
|
if ( !haveRealChannel( comms, forceChannel ) ) {
|
|
|
|
/* See if we have a channel for this address. Then see if we have an
|
|
|
|
invite matching this one, and if not add one. Then trigger a send of
|
|
|
|
it. */
|
|
|
|
|
|
|
|
/* remove the old rec, if found */
|
|
|
|
nukeInvites( comms, xwe, forceChannel );
|
|
|
|
|
2022-12-21 16:11:55 -08:00
|
|
|
XP_U16 flags = COMMS_VERSION;
|
2023-02-15 08:31:51 -08:00
|
|
|
/*AddressRecord* rec = */rememberChannelAddress( comms, forceChannel,
|
2022-12-21 16:11:55 -08:00
|
|
|
0, destAddr, flags );
|
2022-10-01 08:43:10 -07:00
|
|
|
MsgQueueElem* elem = makeInviteElem( comms, xwe, forceChannel, nli );
|
|
|
|
|
2023-02-01 19:08:04 -08:00
|
|
|
elem = addToQueue( comms, xwe, elem, XP_TRUE );
|
2023-03-25 10:23:41 -07:00
|
|
|
if ( !!elem ) {
|
2023-03-18 15:15:34 -07:00
|
|
|
XP_ASSERT( !elem->smp.next );
|
2023-03-25 10:23:41 -07:00
|
|
|
XP_LOGFF( "added invite on channel %d", elem->channelNo & CHANNEL_MASK );
|
|
|
|
/* Let's let platform code decide whether to call sendMsg() . On
|
|
|
|
Android creating a game with an invitation in its queue is always
|
|
|
|
followed by opening the game, which results in comms_resendAll()
|
|
|
|
getting called leading to a second send immediately after this. So
|
|
|
|
let Android drop it. Linux, though, needs it for now. */
|
|
|
|
if ( sendNow && !!elem && !!comms->procs.sendInvt ) {
|
|
|
|
sendMsg( comms, xwe, elem, COMMS_CONN_NONE );
|
|
|
|
}
|
2023-01-30 12:02:46 -08:00
|
|
|
}
|
2022-10-01 08:43:10 -07:00
|
|
|
}
|
|
|
|
LOG_RETURN_VOID();
|
2022-09-06 12:16:29 -07:00
|
|
|
}
|
2022-10-20 12:19:33 -07:00
|
|
|
|
2023-02-01 19:08:04 -08:00
|
|
|
typedef struct _GetInvitedData {
|
|
|
|
XP_U16 allBits;
|
|
|
|
XP_U16 count;
|
|
|
|
} GetInvitedData;
|
2022-10-20 12:19:33 -07:00
|
|
|
|
2023-02-01 19:08:04 -08:00
|
|
|
static ForEachAct
|
|
|
|
getInvitedProc( MsgQueueElem* elem, void* closure )
|
|
|
|
{
|
|
|
|
if ( IS_INVITE( elem ) ) {
|
|
|
|
GetInvitedData* gidp = (GetInvitedData*)closure;
|
|
|
|
XP_PlayerAddr channelNo = elem->channelNo & CHANNEL_MASK;
|
|
|
|
XP_LOGFF( "got invite on channel %d", channelNo );
|
|
|
|
XP_U16 thisBit = 1 << channelNo;
|
|
|
|
XP_ASSERT( 0 == (thisBit & gidp->allBits) ); /* should be no dupes */
|
|
|
|
if ( 0 == (thisBit & gidp->allBits) ) {
|
|
|
|
++gidp->count;
|
2022-10-20 12:19:33 -07:00
|
|
|
}
|
2023-02-01 19:08:04 -08:00
|
|
|
gidp->allBits |= thisBit;
|
2022-10-20 12:19:33 -07:00
|
|
|
}
|
2023-02-01 19:08:04 -08:00
|
|
|
return FEA_OK;
|
|
|
|
}
|
2022-10-20 12:19:33 -07:00
|
|
|
|
2023-02-01 19:08:04 -08:00
|
|
|
void
|
|
|
|
comms_getInvited( const CommsCtxt* comms, XP_U16* nInvites )
|
|
|
|
{
|
|
|
|
GetInvitedData gid = {0};
|
|
|
|
forEachElem( (CommsCtxt*)comms, getInvitedProc, &gid );
|
|
|
|
*nInvites = gid.count;
|
2022-12-10 22:15:33 -08:00
|
|
|
// LOG_RETURNF( "%d", *nInvites );
|
2022-10-20 12:19:33 -07:00
|
|
|
}
|
2022-09-06 12:16:29 -07:00
|
|
|
#endif
|
|
|
|
|
2007-11-18 23:43:27 +00:00
|
|
|
/* Send a message using the sequentially next MsgID. Save the message so
|
|
|
|
* resend can work. */
|
|
|
|
XP_S16
|
2020-04-25 17:47:07 -07:00
|
|
|
comms_send( CommsCtxt* comms, XWEnv xwe, XWStreamCtxt* stream )
|
2007-11-18 23:43:27 +00:00
|
|
|
{
|
2007-11-26 02:58:25 +00:00
|
|
|
XP_S16 result = -1;
|
2023-01-06 07:52:41 -08:00
|
|
|
THREAD_CHECK_START(comms);
|
2012-11-13 07:22:48 -08:00
|
|
|
if ( 0 == stream_getSize(stream) ) {
|
2022-04-29 09:19:44 -07:00
|
|
|
XP_LOGFF( "dropping 0-len message" );
|
2012-11-13 07:22:48 -08:00
|
|
|
} else {
|
|
|
|
XP_PlayerAddr channelNo = stream_getAddress( stream );
|
2014-11-06 18:44:19 -08:00
|
|
|
CNO_FMT( cbuf, channelNo );
|
2022-04-29 09:19:44 -07:00
|
|
|
XP_LOGFF( "%s", cbuf );
|
2023-02-15 08:31:51 -08:00
|
|
|
AddressRecord* rec = getRecordFor( comms, channelNo );
|
2012-11-13 07:22:48 -08:00
|
|
|
MsgID msgID = (!!rec)? ++rec->nextMsgID : 0;
|
|
|
|
MsgQueueElem* elem;
|
|
|
|
|
|
|
|
if ( 0 == channelNo ) {
|
|
|
|
channelNo = comms_getChannelSeed(comms) & ~CHANNEL_MASK;
|
2022-12-27 16:03:08 -08:00
|
|
|
XP_LOGFF( "new channelNo: %X", channelNo );
|
2012-11-13 07:22:48 -08:00
|
|
|
}
|
2007-11-18 23:43:27 +00:00
|
|
|
|
2021-12-15 10:03:09 -08:00
|
|
|
XP_LOGFF( TAGFMT() "assigning msgID=" XP_LD " on %s", TAGPRMS, msgID, cbuf );
|
2007-11-26 02:58:25 +00:00
|
|
|
|
2020-04-25 17:47:07 -07:00
|
|
|
elem = makeElemWithID( comms, xwe, msgID, rec, channelNo, stream );
|
2012-11-13 07:22:48 -08:00
|
|
|
if ( NULL != elem ) {
|
2023-02-01 19:08:04 -08:00
|
|
|
elem = addToQueue( comms, xwe, elem, XP_TRUE );
|
2023-03-25 10:23:41 -07:00
|
|
|
if ( !!elem ) {
|
|
|
|
printQueue( comms );
|
|
|
|
result = sendMsg( comms, xwe, elem, COMMS_CONN_NONE );
|
|
|
|
}
|
2012-11-13 07:22:48 -08:00
|
|
|
}
|
2007-11-26 02:58:25 +00:00
|
|
|
}
|
2023-01-06 07:52:41 -08:00
|
|
|
THREAD_CHECK_END();
|
2007-11-26 02:58:25 +00:00
|
|
|
return result;
|
2003-11-01 05:35:29 +00:00
|
|
|
} /* comms_send */
|
|
|
|
|
2019-12-10 14:42:54 -08:00
|
|
|
static void
|
2020-04-25 17:47:07 -07:00
|
|
|
notifyQueueChanged( const CommsCtxt* comms, XWEnv xwe )
|
2019-12-10 14:42:54 -08:00
|
|
|
{
|
2021-02-07 16:31:32 -08:00
|
|
|
if ( !!comms->procs.countChanged ) {
|
2023-11-06 11:20:00 -08:00
|
|
|
XP_U16 count = comms->queueLen;
|
|
|
|
XP_Bool quashed = QUASHED(comms);
|
|
|
|
(*comms->procs.countChanged)( xwe, comms->procs.closure, count, quashed );
|
2021-02-07 16:31:32 -08:00
|
|
|
}
|
2019-12-10 14:42:54 -08:00
|
|
|
}
|
|
|
|
|
2003-11-01 05:35:29 +00:00
|
|
|
/* 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.
|
|
|
|
*/
|
2013-07-12 07:34:12 -07:00
|
|
|
static MsgQueueElem*
|
2023-02-01 19:08:04 -08:00
|
|
|
addToQueue( CommsCtxt* comms, XWEnv xwe, MsgQueueElem* newElem, XP_Bool notify )
|
2003-11-01 05:35:29 +00:00
|
|
|
{
|
2013-07-12 07:34:12 -07:00
|
|
|
MsgQueueElem* asAdded = newElem;
|
2023-01-06 07:52:41 -08:00
|
|
|
THREAD_CHECK_START( comms );
|
2023-03-18 15:15:34 -07:00
|
|
|
newElem->smp.next = NULL;
|
2023-03-16 07:41:53 -07:00
|
|
|
|
|
|
|
MsgQueueElem** head;
|
|
|
|
AddressRecord* rec = getRecordFor( comms, newElem->channelNo );
|
2023-03-25 10:23:41 -07:00
|
|
|
if ( !rec ) {
|
|
|
|
freeElem( MPPARM(comms->mpool) newElem );
|
|
|
|
asAdded = NULL;
|
|
|
|
goto dropPacket;
|
|
|
|
}
|
2023-03-16 07:41:53 -07:00
|
|
|
head = &rec->_msgQueueHead;
|
|
|
|
|
|
|
|
if ( !*head ) {
|
|
|
|
*head = newElem;
|
2003-11-01 05:35:29 +00:00
|
|
|
} else {
|
2023-03-18 15:15:34 -07:00
|
|
|
while ( !!(*head)->smp.next ) {
|
|
|
|
head = (MsgQueueElem**)&(*head)->smp.next;
|
2023-02-01 19:08:04 -08:00
|
|
|
}
|
2023-03-16 07:41:53 -07:00
|
|
|
if ( elems_same( *head, newElem ) ) {
|
2023-02-01 19:08:04 -08:00
|
|
|
/* This does still happen! Not sure why. */
|
2022-09-28 07:24:45 -07:00
|
|
|
freeElem( MPPARM(comms->mpool) newElem );
|
2023-03-16 07:41:53 -07:00
|
|
|
asAdded = *head;
|
2013-07-12 07:34:12 -07:00
|
|
|
} else {
|
2023-03-18 15:15:34 -07:00
|
|
|
(*head)->smp.next = &newElem->smp;
|
2013-07-12 07:34:12 -07:00
|
|
|
}
|
2003-11-01 05:35:29 +00:00
|
|
|
|
|
|
|
XP_ASSERT( comms->queueLen > 0 );
|
|
|
|
}
|
2013-07-12 07:34:12 -07:00
|
|
|
|
|
|
|
if ( newElem == asAdded ) {
|
|
|
|
++comms->queueLen;
|
2023-03-25 10:13:27 -07:00
|
|
|
/* Do I need this? PENDING */
|
|
|
|
formatMsgNo( comms, newElem, (XP_UCHAR*)newElem->smp.msgNo,
|
|
|
|
sizeof(newElem->smp.msgNo) );
|
2023-02-01 19:08:04 -08:00
|
|
|
if ( notify ) {
|
|
|
|
notifyQueueChanged( comms, xwe );
|
|
|
|
}
|
2013-07-12 07:34:12 -07:00
|
|
|
}
|
2023-03-25 10:23:41 -07:00
|
|
|
dropPacket:
|
2013-07-05 21:32:41 -07:00
|
|
|
XP_ASSERT( comms->queueLen <= 128 ); /* reasonable limit in testing */
|
2023-01-06 07:52:41 -08:00
|
|
|
THREAD_CHECK_END();
|
2013-07-12 07:34:12 -07:00
|
|
|
return asAdded;
|
2003-11-01 05:35:29 +00:00
|
|
|
} /* addToQueue */
|
|
|
|
|
|
|
|
#ifdef DEBUG
|
2023-02-01 19:08:04 -08:00
|
|
|
static ForEachAct
|
|
|
|
printElem( MsgQueueElem* elem, void* closure )
|
2003-11-01 05:35:29 +00:00
|
|
|
{
|
2023-02-01 19:08:04 -08:00
|
|
|
int* iip = (int*)closure;
|
|
|
|
CNO_FMT( cbuf, elem->channelNo );
|
|
|
|
XP_LOGFF( "%d: %s; msgID=" XP_LD
|
2010-10-18 22:08:04 -07:00
|
|
|
#ifdef COMMS_CHECKSUM
|
2023-02-01 19:08:04 -08:00
|
|
|
"; sum=%s"
|
2010-10-18 22:08:04 -07:00
|
|
|
#endif
|
2023-02-01 19:08:04 -08:00
|
|
|
, *iip, cbuf, elem->msgID
|
2010-10-18 22:08:04 -07:00
|
|
|
#ifdef COMMS_CHECKSUM
|
2023-02-01 19:08:04 -08:00
|
|
|
, elem->checksum
|
2010-10-18 22:08:04 -07:00
|
|
|
#endif
|
2023-02-01 19:08:04 -08:00
|
|
|
);
|
|
|
|
++*iip;
|
|
|
|
return FEA_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
printQueue( const CommsCtxt* comms )
|
|
|
|
{
|
|
|
|
int ii = 0;
|
|
|
|
forEachElem( (CommsCtxt*)comms, printElem, &ii );
|
2003-11-01 05:35:29 +00:00
|
|
|
}
|
2009-02-28 15:06:12 +00:00
|
|
|
|
|
|
|
static void
|
2023-03-27 14:29:49 -07:00
|
|
|
_assertQueueOk( const CommsCtxt* comms, const char* XP_UNUSED(func) )
|
2009-02-28 15:06:12 +00:00
|
|
|
{
|
|
|
|
XP_U16 count = 0;
|
2023-03-16 07:41:53 -07:00
|
|
|
|
|
|
|
for ( AddressRecord* recs = comms->recs; !!recs; recs = recs->next ) {
|
2023-03-18 15:15:34 -07:00
|
|
|
for ( MsgQueueElem* elem = recs->_msgQueueHead; !!elem;
|
|
|
|
elem = (MsgQueueElem*)elem->smp.next ) {
|
2023-03-16 07:41:53 -07:00
|
|
|
++count;
|
|
|
|
}
|
|
|
|
}
|
2023-03-25 10:23:41 -07:00
|
|
|
if ( count != comms->queueLen ) {
|
|
|
|
XP_LOGFF( "count(%d) != comms->queueLen(%d)", count, comms->queueLen );
|
|
|
|
XP_ASSERT(0);
|
|
|
|
}
|
2011-12-15 18:37:20 -08:00
|
|
|
if ( count >= 10 ) {
|
2022-04-29 09:19:44 -07:00
|
|
|
XP_LOGFF( "queueLen unexpectedly high: %d", count );
|
2011-12-15 18:37:20 -08:00
|
|
|
}
|
2009-02-28 15:06:12 +00:00
|
|
|
}
|
2022-09-03 18:47:04 -07:00
|
|
|
|
|
|
|
static void
|
|
|
|
assertAddrOk( const CommsAddrRec* addr )
|
|
|
|
{
|
|
|
|
CommsConnType typ;
|
|
|
|
for ( XP_U32 st = 0; addr_iter( addr, &typ, &st ); ) {
|
|
|
|
switch ( typ ) {
|
|
|
|
case COMMS_CONN_MQTT:
|
|
|
|
XP_ASSERT( 0 != addr->u.mqtt.devID );
|
|
|
|
break;
|
|
|
|
case COMMS_CONN_SMS:
|
|
|
|
XP_ASSERT( 0 != addr->u.sms.phone[0] );
|
|
|
|
break;
|
2022-09-04 09:59:47 -07:00
|
|
|
case COMMS_CONN_P2P:
|
|
|
|
XP_ASSERT( 0 != addr->u.p2p.mac_addr[0] );
|
|
|
|
break;
|
2022-09-06 20:38:56 -07:00
|
|
|
case COMMS_CONN_BT:
|
|
|
|
/* XP_ASSERT( 0 != addr->u.bt.btAddr.chars[0] */
|
|
|
|
/* || 0 != addr->u.bt.hostName[0] ); */
|
|
|
|
break;
|
|
|
|
case COMMS_CONN_RELAY:
|
|
|
|
case COMMS_CONN_NFC:
|
|
|
|
break;
|
2022-09-03 18:47:04 -07:00
|
|
|
default:
|
2022-09-04 09:59:47 -07:00
|
|
|
XP_LOGFF( "no case for %s", ConnType2Str(typ) );
|
2022-09-03 18:47:04 -07:00
|
|
|
XP_ASSERT(0);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2013-07-16 08:17:46 -07:00
|
|
|
#endif
|
2013-07-12 07:34:12 -07:00
|
|
|
|
|
|
|
static XP_Bool
|
|
|
|
elems_same( const MsgQueueElem* elem1, const MsgQueueElem* elem2 )
|
|
|
|
{
|
|
|
|
XP_Bool same = elem1->msgID == elem2->msgID
|
|
|
|
&& elem1->channelNo == elem2->channelNo
|
2023-03-18 15:15:34 -07:00
|
|
|
&& elem1->smp.len == elem2->smp.len
|
|
|
|
&& 0 == XP_MEMCMP( elem1->smp.buf, elem2->smp.buf, elem1->smp.len );
|
2013-07-12 07:34:12 -07:00
|
|
|
return same;
|
|
|
|
}
|
2006-09-17 05:06:46 +00:00
|
|
|
|
2007-11-26 02:58:25 +00:00
|
|
|
static void
|
2022-09-28 07:24:45 -07:00
|
|
|
freeElem( MPFORMAL MsgQueueElem* elem )
|
2007-11-26 02:58:25 +00:00
|
|
|
{
|
2023-03-18 15:15:34 -07:00
|
|
|
XP_ASSERT( !elem->smp.next );
|
|
|
|
XP_FREEP( mpool, &elem->smp.buf );
|
2010-10-18 22:08:04 -07:00
|
|
|
#ifdef COMMS_CHECKSUM
|
2023-03-18 15:15:34 -07:00
|
|
|
/* XP_LOGFF( "freeing msg with len %d, sum %s", elem->smp.len, elem->checksum ); */
|
2023-02-01 19:08:04 -08:00
|
|
|
XP_FREEP( mpool, &elem->checksum );
|
2022-12-26 19:24:39 -08:00
|
|
|
#else
|
2023-03-18 15:15:34 -07:00
|
|
|
/* XP_LOGFF( "freeing msg with len %d", elem->smp.len ); */
|
2010-10-18 20:17:44 -07:00
|
|
|
#endif
|
2022-09-28 07:24:45 -07:00
|
|
|
XP_FREE( mpool, elem );
|
2007-11-26 02:58:25 +00:00
|
|
|
}
|
|
|
|
|
2003-11-01 05:35:29 +00: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 05:49:45 +00: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 05:35:29 +00:00
|
|
|
*/
|
2023-02-01 19:08:04 -08:00
|
|
|
|
|
|
|
typedef struct _RemoveData {
|
|
|
|
const CommsCtxt* comms;
|
|
|
|
XP_PlayerAddr channelNo;
|
|
|
|
MsgID msgID;
|
|
|
|
} RemoveData;
|
|
|
|
|
|
|
|
static ForEachAct
|
|
|
|
removeProc( MsgQueueElem* elem, void* closure )
|
|
|
|
{
|
|
|
|
ForEachAct result = FEA_OK;
|
|
|
|
RemoveData* rdp = (RemoveData*)closure;
|
|
|
|
|
|
|
|
XP_PlayerAddr maskedChannelNo = ~CHANNEL_MASK & rdp->channelNo;
|
|
|
|
XP_Bool knownGood = XP_FALSE;
|
|
|
|
/* 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. */
|
|
|
|
|
|
|
|
XP_PlayerAddr maskedElemChannelNo = ~CHANNEL_MASK & elem->channelNo;
|
|
|
|
if ( (maskedElemChannelNo == 0) && (rdp->channelNo != 0) ) {
|
|
|
|
XP_ASSERT( !rdp->comms->isServer || IS_INVITE(elem) );
|
|
|
|
XP_ASSERT( elem->msgID == 0 );
|
|
|
|
} else if ( maskedElemChannelNo != maskedChannelNo ) {
|
|
|
|
knownGood = XP_TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
if ( !knownGood && (elem->msgID <= rdp->msgID) ) {
|
|
|
|
result = FEA_REMOVE;
|
|
|
|
}
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
2003-11-01 05:35:29 +00:00
|
|
|
static void
|
2020-04-25 17:47:07 -07:00
|
|
|
removeFromQueue( CommsCtxt* comms, XWEnv xwe, XP_PlayerAddr channelNo, MsgID msgID )
|
2003-11-01 05:35:29 +00:00
|
|
|
{
|
2023-01-06 07:52:41 -08:00
|
|
|
THREAD_CHECK_START( comms );
|
2023-03-16 07:41:53 -07:00
|
|
|
assertQueueOk( comms );
|
2014-11-06 18:44:19 -08:00
|
|
|
CNO_FMT( cbuf, channelNo );
|
2022-04-29 09:19:44 -07:00
|
|
|
XP_LOGFF( "(channelNo=%d): remove msgs <= " XP_LD " for %s (queueLen: %d)",
|
|
|
|
channelNo, msgID, cbuf, comms->queueLen );
|
2022-12-30 16:14:22 -08:00
|
|
|
#ifdef DEBUG
|
2022-12-26 19:24:39 -08:00
|
|
|
XP_U16 prevLen = comms->queueLen;
|
2022-12-30 16:14:22 -08:00
|
|
|
#endif
|
2022-12-26 19:24:39 -08:00
|
|
|
|
2023-02-15 08:31:51 -08:00
|
|
|
if ((channelNo == 0) || !!getRecordFor( comms, channelNo)) {
|
2009-02-28 14:40:50 +00:00
|
|
|
|
2023-02-01 19:08:04 -08:00
|
|
|
RemoveData rd = {
|
|
|
|
.comms = comms,
|
|
|
|
.msgID = msgID,
|
|
|
|
.channelNo = channelNo,
|
|
|
|
};
|
|
|
|
forEachElem( comms, removeProc, &rd );
|
2006-09-15 07:33:59 +00:00
|
|
|
|
2020-04-25 17:47:07 -07:00
|
|
|
notifyQueueChanged( comms, xwe );
|
2003-11-01 05:35:29 +00:00
|
|
|
}
|
2006-09-27 01:54:53 +00:00
|
|
|
|
2022-12-26 19:24:39 -08:00
|
|
|
XP_ASSERT( comms->queueLen <= prevLen );
|
|
|
|
XP_LOGFF( "queueLen now %d (was %d)", comms->queueLen, prevLen );
|
2003-11-01 05:35:29 +00:00
|
|
|
|
2006-09-17 05:06:46 +00:00
|
|
|
#ifdef DEBUG
|
2009-02-28 15:06:12 +00:00
|
|
|
assertQueueOk( comms );
|
2003-11-01 05:35:29 +00:00
|
|
|
printQueue( comms );
|
2006-09-17 05:06:46 +00:00
|
|
|
#endif
|
2023-01-06 07:52:41 -08:00
|
|
|
THREAD_CHECK_END();
|
2003-11-01 05:35:29 +00:00
|
|
|
} /* removeFromQueue */
|
|
|
|
|
2012-01-25 18:27:37 -08:00
|
|
|
static XP_U32
|
|
|
|
gameID( const CommsCtxt* comms )
|
|
|
|
{
|
|
|
|
XP_U32 gameID = comms->connID;
|
|
|
|
if ( 0 == gameID ) {
|
|
|
|
gameID = comms->util->gameInfo->gameID;
|
|
|
|
}
|
2013-05-30 06:48:29 -07:00
|
|
|
|
2012-01-25 18:27:37 -08:00
|
|
|
/* Most of the time these will be the same, but early in a game they won't
|
|
|
|
be. Would be nice not to have to use gameID. */
|
2013-05-30 06:48:29 -07:00
|
|
|
if ( 0 == gameID ) {
|
2022-04-29 09:19:44 -07:00
|
|
|
XP_LOGFF( "gameID STILL 0" );
|
2013-05-30 06:48:29 -07:00
|
|
|
} else if ( 0 == comms->util->gameInfo->gameID ) {
|
2022-04-29 09:19:44 -07:00
|
|
|
XP_LOGFF( "setting gi's gameID to 0X%X", gameID );
|
2013-05-30 06:48:29 -07:00
|
|
|
comms->util->gameInfo->gameID = gameID;
|
|
|
|
}
|
|
|
|
|
2012-01-25 18:27:37 -08:00
|
|
|
return gameID;
|
|
|
|
}
|
|
|
|
|
2023-03-13 12:25:08 -07:00
|
|
|
#ifdef DEBUG
|
|
|
|
typedef struct _CheckPrevState {
|
|
|
|
const CommsCtxt* comms;
|
|
|
|
const MsgQueueElem* elem;
|
|
|
|
int count;
|
|
|
|
} CheckPrevState;
|
|
|
|
|
|
|
|
static ForEachAct
|
|
|
|
checkPrevProc( MsgQueueElem* elem, void* closure )
|
|
|
|
{
|
|
|
|
CheckPrevState* cpsp = (CheckPrevState*)closure;
|
|
|
|
if ( elem != cpsp->elem
|
|
|
|
&& (cpsp->elem->channelNo & CHANNEL_MASK) == (elem->channelNo & CHANNEL_MASK) ) {
|
|
|
|
if ( 0 == cpsp->elem->msgID || elem->msgID < cpsp->elem->msgID ) {
|
|
|
|
++cpsp->count;
|
|
|
|
XP_LOGFF( "found one! their id: %d; my id: %d", elem->msgID,
|
|
|
|
cpsp->elem->msgID );
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return FEA_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
checkForPrev( const CommsCtxt* comms, MsgQueueElem* elem, CommsConnType typ )
|
|
|
|
{
|
|
|
|
if ( COMMS_CONN_MQTT == typ ) {
|
|
|
|
CheckPrevState cps = { .comms = comms,
|
2023-10-26 12:25:59 -07:00
|
|
|
.elem = elem,
|
2023-03-13 12:25:08 -07:00
|
|
|
};
|
|
|
|
forEachElem( (CommsCtxt*)comms, checkPrevProc, &cps );
|
|
|
|
}
|
|
|
|
}
|
|
|
|
#else
|
|
|
|
# define checkForPrev( comms, elem, typ )
|
|
|
|
#endif
|
|
|
|
|
2003-11-01 05:35:29 +00:00
|
|
|
static XP_S16
|
2023-01-06 07:52:41 -08:00
|
|
|
sendMsg( const CommsCtxt* comms, XWEnv xwe, MsgQueueElem* elem,
|
|
|
|
const CommsConnType filter )
|
2003-11-01 05:35:29 +00:00
|
|
|
{
|
2007-11-26 02:58:25 +00:00
|
|
|
XP_S16 result = -1;
|
2014-11-11 07:46:01 -08:00
|
|
|
XP_PlayerAddr channelNo = elem->channelNo;
|
|
|
|
CNO_FMT( cbuf, channelNo );
|
2003-11-01 05:35:29 +00:00
|
|
|
|
2022-09-08 09:57:11 -07:00
|
|
|
XP_Bool isInvite = IS_INVITE(elem);
|
2013-06-30 08:33:29 -07:00
|
|
|
#ifdef COMMS_CHECKSUM
|
2022-09-06 12:16:29 -07:00
|
|
|
XP_LOGFF( TAGFMT() "sending message on %s: id: %d; len: %d; sum: %s; isInvite: %s",
|
2023-03-18 15:15:34 -07:00
|
|
|
TAGPRMS, cbuf, elem->msgID, elem->smp.len, elem->checksum, boolToStr(isInvite) );
|
2013-06-30 08:33:29 -07:00
|
|
|
#endif
|
|
|
|
|
2022-09-04 09:59:47 -07:00
|
|
|
const CommsAddrRec* addrP = NULL;
|
2022-09-03 18:47:04 -07:00
|
|
|
if ( comms->isServer ) {
|
2023-02-15 08:31:51 -08:00
|
|
|
(void)channelToAddress( comms, channelNo, &addrP );
|
2022-09-03 18:47:04 -07:00
|
|
|
} else {
|
|
|
|
/* guest has only one peer, but old code might save several */
|
|
|
|
if ( !!comms->recs ) {
|
2022-09-06 17:50:41 -07:00
|
|
|
XP_ASSERT( !comms->recs->next ); // firing during upgrade test
|
2022-09-03 18:47:04 -07:00
|
|
|
addrP = &comms->recs->addr;
|
|
|
|
}
|
|
|
|
}
|
2023-11-05 11:52:48 -08:00
|
|
|
if ( QUASHED(comms) ) {
|
|
|
|
// XP_LOGFF( "not sending; comms is quashed" );
|
|
|
|
} else if ( NULL == addrP ) {
|
2022-09-03 18:47:04 -07:00
|
|
|
XP_LOGFF( TAGFMT() "no addr for channel %x; dropping!'", TAGPRMS, channelNo );
|
2022-09-28 07:24:45 -07:00
|
|
|
// XP_ASSERT(0); /* firing */
|
2020-02-24 21:10:49 +01:00
|
|
|
} else {
|
2022-09-03 18:47:04 -07:00
|
|
|
CommsAddrRec addr = *addrP;
|
2022-09-03 19:14:40 -07:00
|
|
|
if ( addr_hasType( &comms->selfAddr, COMMS_CONN_NFC ) ) {
|
2020-03-24 15:58:06 -07:00
|
|
|
addr_addType( &addr, COMMS_CONN_NFC );
|
|
|
|
}
|
2020-02-24 21:10:49 +01:00
|
|
|
|
2022-09-03 18:47:04 -07:00
|
|
|
CommsConnType typ;
|
|
|
|
for ( XP_U32 st = 0; addr_iter( &addr, &typ, &st ); ) {
|
|
|
|
XP_S16 nSent = -1;
|
|
|
|
if ( comms_getAddrDisabled( comms, typ, XP_TRUE ) ) {
|
|
|
|
XP_LOGFF( "dropping message because %s disabled",
|
|
|
|
ConnType2Str( typ ) );
|
|
|
|
} else if ( COMMS_CONN_NONE != filter && filter != typ ) {
|
|
|
|
XP_LOGFF( "dropping message because not of type %s",
|
|
|
|
ConnType2Str( filter ) );
|
|
|
|
} else {
|
2023-01-23 13:42:38 -08:00
|
|
|
if ( !isInvite && !addr_hasType( &comms->selfAddr, typ ) ) {
|
|
|
|
XP_LOGFF( "self addr doesn't have msg type %s", ConnType2Str(typ) );
|
2023-01-24 11:38:11 -08:00
|
|
|
/* PENDING: fix this */
|
|
|
|
// XP_ASSERT( 0 ); <-- happens a lot
|
2023-01-23 13:42:38 -08:00
|
|
|
}
|
2021-02-07 16:31:32 -08:00
|
|
|
#ifdef COMMS_CHECKSUM
|
2022-09-03 18:47:04 -07:00
|
|
|
XP_LOGFF( TAGFMT() "sending msg with sum %s using typ %s", TAGPRMS,
|
|
|
|
elem->checksum, ConnType2Str(typ) );
|
2021-02-07 16:31:32 -08:00
|
|
|
#endif
|
2022-09-03 18:47:04 -07:00
|
|
|
switch ( typ ) {
|
2006-10-10 01:34:37 +00:00
|
|
|
#ifdef XWFEATURE_RELAY
|
2022-09-03 18:47:04 -07:00
|
|
|
case COMMS_CONN_RELAY: {
|
|
|
|
XWHostID destID = getDestID( comms, xwe, channelNo );
|
|
|
|
if ( HOST_ID_NONE == destID ) {
|
|
|
|
XP_LOGFF( TAGFMT() "skipping message via relay: no destID yet", TAGPRMS );
|
|
|
|
} else if ( haveRelayID( comms ) && sendNoConn( comms, xwe, elem, destID ) ) {
|
|
|
|
/* do nothing */
|
2023-03-18 15:15:34 -07:00
|
|
|
nSent = elem->smp.len;
|
2022-09-03 18:47:04 -07:00
|
|
|
} else if ( comms->rr.relayState >= COMMS_RELAYSTATE_CONNECTED ) {
|
|
|
|
XP_UCHAR msgNo[16];
|
|
|
|
formatMsgNo( comms, elem, msgNo, sizeof(msgNo) );
|
|
|
|
if ( send_via_relay( comms, xwe, XWRELAY_MSG_TORELAY, destID,
|
2023-03-18 15:15:34 -07:00
|
|
|
elem->smp.buf, elem->smp.len, msgNo ) ) {
|
|
|
|
nSent = elem->smp.len;
|
2022-09-03 18:47:04 -07:00
|
|
|
}
|
|
|
|
} else {
|
|
|
|
XP_LOGFF( "skipping message: not connected to relay" );
|
2014-11-06 06:35:28 -08:00
|
|
|
}
|
2022-09-03 18:47:04 -07:00
|
|
|
break;
|
2014-10-15 07:26:18 -07:00
|
|
|
}
|
2007-02-08 02:53:10 +00:00
|
|
|
#endif
|
2012-02-07 17:38:24 -08:00
|
|
|
#if defined XWFEATURE_IP_DIRECT
|
2022-09-03 18:47:04 -07:00
|
|
|
case COMMS_CONN_BT:
|
|
|
|
case COMMS_CONN_IP_DIRECT:
|
|
|
|
nSent = send_via_ip( comms, BTIPMSG_DATA, channelNo,
|
2023-03-18 15:15:34 -07:00
|
|
|
elem->smp.buf, elem->smp.len );
|
2007-11-26 02:58:25 +00:00
|
|
|
#ifdef COMMS_HEARTBEAT
|
2022-09-03 18:47:04 -07:00
|
|
|
setHeartbeatTimer( comms );
|
2007-11-26 02:58:25 +00:00
|
|
|
#endif
|
2022-09-03 18:47:04 -07:00
|
|
|
break;
|
2005-07-23 15:28:15 +00:00
|
|
|
#endif
|
2022-10-01 08:43:10 -07:00
|
|
|
default:
|
2022-09-03 18:47:04 -07:00
|
|
|
XP_ASSERT( addr_hasType( &addr, typ ) );
|
2005-03-19 22:01:38 +00:00
|
|
|
|
2022-09-03 18:47:04 -07:00
|
|
|
/* A more general check that the address type has the settings
|
|
|
|
it needs would be better here.... */
|
|
|
|
if ( typ == COMMS_CONN_MQTT && 0 == addr.u.mqtt.devID ) {
|
|
|
|
XP_LOGFF( "not sending: MQTT address NULL" );
|
|
|
|
XP_ASSERT(0);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2022-09-06 13:58:36 -07:00
|
|
|
if ( 0 ) {
|
|
|
|
#ifdef XWFEATURE_COMMS_INVITE
|
2022-10-01 08:43:10 -07:00
|
|
|
} else if ( isInvite ) {
|
|
|
|
if ( !!comms->procs.sendInvt ) {
|
2023-10-24 19:51:54 -07:00
|
|
|
NetLaunchInfo nli;
|
|
|
|
XP_MEMCPY( &nli, elem->smp.buf, sizeof(nli) );
|
2023-03-18 15:15:34 -07:00
|
|
|
XP_ASSERT( 0 != elem->smp.createdStamp );
|
2023-10-24 19:51:54 -07:00
|
|
|
nSent = (*comms->procs.sendInvt)( xwe, &nli,
|
2023-03-18 15:15:34 -07:00
|
|
|
elem->smp.createdStamp,
|
|
|
|
&addr, typ,
|
|
|
|
comms->procs.closure );
|
2022-10-01 08:43:10 -07:00
|
|
|
}
|
2022-09-06 13:58:36 -07:00
|
|
|
#endif
|
|
|
|
} else {
|
2023-03-26 10:36:58 -07:00
|
|
|
SendMsgsPacket* head = NULL;
|
2023-03-25 10:13:27 -07:00
|
|
|
if ( COMMS_CONN_MQTT == typ ) {
|
|
|
|
AddressRecord* rec = getRecordFor( comms, channelNo);
|
|
|
|
head = &rec->_msgQueueHead->smp;
|
2023-10-26 12:25:59 -07:00
|
|
|
#ifdef DEBUG
|
|
|
|
/* Make sure our message is in there!!! */
|
|
|
|
XP_Bool found = XP_FALSE;
|
|
|
|
for ( SendMsgsPacket* tmp = head; !found && !!tmp; tmp = tmp->next ) {
|
|
|
|
found = tmp == &elem->smp;
|
|
|
|
}
|
|
|
|
XP_ASSERT( found );
|
|
|
|
#endif
|
2023-03-25 10:13:27 -07:00
|
|
|
} else {
|
2023-03-26 10:36:58 -07:00
|
|
|
XP_ASSERT( !elem->smp.next );
|
|
|
|
}
|
|
|
|
if ( !head ) {
|
2023-03-25 10:13:27 -07:00
|
|
|
head = &elem->smp;
|
|
|
|
}
|
2023-03-27 14:29:49 -07:00
|
|
|
XP_ASSERT( !!head );
|
2023-03-17 19:21:57 -07:00
|
|
|
XP_ASSERT( !!comms->procs.sendMsgs );
|
2022-09-06 13:58:36 -07:00
|
|
|
XP_U32 gameid = gameID( comms );
|
2023-02-16 21:36:46 -08:00
|
|
|
logAddrComms( comms, &addr, __func__ );
|
2023-03-18 15:15:34 -07:00
|
|
|
XP_ASSERT( 0 != elem->smp.createdStamp );
|
2023-03-25 10:13:27 -07:00
|
|
|
nSent = (*comms->procs.sendMsgs)( xwe, head, comms->streamVersion,
|
|
|
|
&addr, typ, gameid,
|
2023-03-17 19:21:57 -07:00
|
|
|
comms->procs.closure );
|
2023-03-13 12:25:08 -07:00
|
|
|
checkForPrev( comms, elem, typ );
|
2022-09-06 13:58:36 -07:00
|
|
|
}
|
2022-10-01 08:43:10 -07:00
|
|
|
break;
|
2022-09-06 13:58:36 -07:00
|
|
|
|
|
|
|
} /* switch */
|
2022-10-01 08:43:10 -07:00
|
|
|
}
|
2022-09-06 13:58:36 -07:00
|
|
|
if ( nSent > result ) {
|
|
|
|
result = nSent;
|
2014-11-06 06:35:28 -08:00
|
|
|
}
|
2022-09-06 13:58:36 -07:00
|
|
|
} /* for */
|
2007-11-26 02:58:25 +00:00
|
|
|
|
2023-03-18 15:15:34 -07:00
|
|
|
if ( result == elem->smp.len ) {
|
2012-11-13 21:18:52 -08:00
|
|
|
#ifdef DEBUG
|
2022-09-03 18:47:04 -07:00
|
|
|
++elem->sendCount;
|
2012-11-13 21:18:52 -08:00
|
|
|
#endif
|
2022-09-03 18:47:04 -07:00
|
|
|
XP_LOGFF( "elem's sendCount since load: %d",
|
|
|
|
elem->sendCount );
|
|
|
|
}
|
|
|
|
CNO_FMT( cbuf1, elem->channelNo );
|
|
|
|
XP_LOGFF( "(%s; msgID=" XP_LD ", len=%d)=>%d", cbuf1, elem->msgID,
|
2023-03-18 15:15:34 -07:00
|
|
|
elem->smp.len, result );
|
2007-11-26 02:58:25 +00:00
|
|
|
}
|
2003-11-01 05:35:29 +00:00
|
|
|
return result;
|
|
|
|
} /* sendMsg */
|
|
|
|
|
2021-03-19 18:51:34 -07:00
|
|
|
#ifdef XWFEATURE_RELAY
|
2010-09-14 13:54:52 -07:00
|
|
|
static void
|
2021-03-19 18:51:34 -07:00
|
|
|
send_relay_ack( CommsCtxt* comms, XWEnv xwe )
|
2010-09-14 13:54:52 -07:00
|
|
|
{
|
2010-09-20 22:06:35 -07:00
|
|
|
LOG_FUNC();
|
2020-04-25 17:47:07 -07:00
|
|
|
(void)send_via_relay( comms, xwe, XWRELAY_ACK, comms->rr.myHostID,
|
|
|
|
NULL, 0, NULL );
|
2010-09-14 13:54:52 -07:00
|
|
|
}
|
2021-03-19 18:51:34 -07:00
|
|
|
#endif
|
2010-09-14 13:54:52 -07:00
|
|
|
|
2023-03-08 20:48:28 -08:00
|
|
|
XP_S16
|
|
|
|
comms_resendAll( CommsCtxt* comms, XWEnv xwe, CommsConnType filter, XP_Bool force )
|
2003-11-01 05:35:29 +00:00
|
|
|
{
|
2015-08-13 07:05:06 -07:00
|
|
|
XP_S16 count = 0;
|
2006-11-11 22:37:36 +00:00
|
|
|
XP_ASSERT( !!comms );
|
|
|
|
|
2020-04-26 11:28:18 -07:00
|
|
|
XP_U32 now = dutil_getCurSeconds( comms->dutil, xwe );
|
2023-11-05 11:52:48 -08:00
|
|
|
if ( QUASHED(comms) ) {
|
|
|
|
// XP_LOGFF( "not sending; comms is quashed" );
|
|
|
|
} else if ( !force && (now < comms->nextResend) ) {
|
2020-10-23 16:37:28 -07:00
|
|
|
XP_LOGFF( "aborting: %d seconds left in backoff",
|
2012-11-13 21:25:03 -08:00
|
|
|
comms->nextResend - now );
|
2023-02-01 19:08:04 -08:00
|
|
|
} else {
|
2023-03-25 10:13:27 -07:00
|
|
|
XP_U32 gameid = gameID( comms );
|
|
|
|
for ( AddressRecord* rec = comms->recs; !!rec; rec = rec->next ) {
|
2023-03-27 14:29:49 -07:00
|
|
|
const MsgQueueElem* const elem = rec->_msgQueueHead;
|
|
|
|
const SendMsgsPacket* const head = &elem->smp;
|
2023-03-25 15:30:09 -07:00
|
|
|
if ( !!head ) {
|
|
|
|
CommsConnType typ;
|
|
|
|
for ( XP_U32 st = 0; addr_iter( &rec->addr, &typ, &st ); ) {
|
|
|
|
if ( COMMS_CONN_NONE == filter || typ == filter ) {
|
2023-03-27 14:29:49 -07:00
|
|
|
if ( IS_INVITE(elem) ) {
|
2023-10-24 19:51:54 -07:00
|
|
|
NetLaunchInfo nli;
|
|
|
|
XP_MEMCPY( &nli, head->buf, sizeof(nli) );
|
|
|
|
(void)(*comms->procs.sendInvt)( xwe, &nli,
|
2023-03-27 14:29:49 -07:00
|
|
|
head->createdStamp,
|
|
|
|
&rec->addr, typ,
|
|
|
|
comms->procs.closure );
|
|
|
|
++count;
|
|
|
|
XP_ASSERT( !head->next );
|
|
|
|
} else {
|
|
|
|
count += (*comms->procs.sendMsgs)( xwe, head, comms->streamVersion,
|
|
|
|
&rec->addr, typ, gameid,
|
|
|
|
comms->procs.closure );
|
|
|
|
}
|
2023-03-25 15:30:09 -07:00
|
|
|
}
|
2023-03-25 10:13:27 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2012-11-13 21:25:03 -08:00
|
|
|
|
|
|
|
/* Now set resend values */
|
2023-03-25 10:13:27 -07:00
|
|
|
comms->resendBackoff = 2 * (1 + comms->resendBackoff);
|
|
|
|
XP_LOGFF( "backoff now %d", comms->resendBackoff );
|
|
|
|
comms->nextResend = now + comms->resendBackoff;
|
2003-11-01 05:35:29 +00:00
|
|
|
}
|
2021-12-15 10:03:09 -08:00
|
|
|
XP_LOGFF( TAGFMT() "=> %d", TAGPRMS, count );
|
2015-08-13 07:05:06 -07:00
|
|
|
return count;
|
2019-11-18 09:26:46 -08:00
|
|
|
}
|
|
|
|
|
2012-04-13 20:21:59 -07:00
|
|
|
#ifdef XWFEATURE_COMMSACK
|
2021-03-11 21:02:48 -08:00
|
|
|
static void
|
|
|
|
ackAnyImpl( CommsCtxt* comms, XWEnv xwe, XP_Bool force )
|
2012-04-13 20:21:59 -07:00
|
|
|
{
|
2023-01-06 07:52:41 -08:00
|
|
|
THREAD_CHECK_START(comms);
|
2013-07-03 07:39:05 -07:00
|
|
|
if ( CONN_ID_NONE == comms->connID ) {
|
2021-03-11 21:02:48 -08:00
|
|
|
XP_LOGFF( "doing nothing because connID still unset" );
|
2013-07-03 07:39:05 -07:00
|
|
|
} else {
|
2022-10-01 08:43:10 -07:00
|
|
|
#ifdef DEBUG
|
|
|
|
int nSent = 0;
|
|
|
|
int nSeen = 0;
|
|
|
|
#endif
|
2023-03-13 12:25:08 -07:00
|
|
|
for ( AddressRecord* rec = comms->recs; !!rec; rec = rec->next ) {
|
2022-10-01 08:43:10 -07:00
|
|
|
#ifdef DEBUG
|
|
|
|
++nSeen;
|
|
|
|
#endif
|
2021-03-11 21:02:48 -08:00
|
|
|
if ( force || rec->lastMsgAckd < rec->lastMsgRcd ) {
|
2022-10-01 08:43:10 -07:00
|
|
|
#ifdef DEBUG
|
2013-07-03 07:39:05 -07:00
|
|
|
++nSent;
|
2014-11-06 18:44:19 -08:00
|
|
|
CNO_FMT( cbuf, rec->channelNo );
|
2021-03-11 21:02:48 -08:00
|
|
|
XP_LOGFF( "%s; %d < %d (or force: %s): rec getting ack",
|
|
|
|
cbuf, rec->lastMsgAckd, rec->lastMsgRcd,
|
|
|
|
boolToStr(force) );
|
2022-10-01 08:43:10 -07:00
|
|
|
#endif
|
2020-04-25 17:47:07 -07:00
|
|
|
sendEmptyMsg( comms, xwe, rec );
|
2013-07-03 07:39:05 -07:00
|
|
|
}
|
2012-04-13 20:21:59 -07:00
|
|
|
}
|
2023-03-25 10:13:27 -07:00
|
|
|
XP_LOGFF( "sent for %d channels (of %d)", nSent, nSeen );
|
2023-01-06 07:52:41 -08:00
|
|
|
}
|
|
|
|
THREAD_CHECK_END();
|
2012-04-13 20:21:59 -07:00
|
|
|
}
|
2021-03-11 21:02:48 -08:00
|
|
|
|
|
|
|
void
|
|
|
|
comms_ackAny( CommsCtxt* comms, XWEnv xwe )
|
|
|
|
{
|
|
|
|
ackAnyImpl( comms, xwe, XP_FALSE );
|
|
|
|
}
|
|
|
|
#else
|
|
|
|
# define ackAnyImpl( comms, xwe, force )
|
2012-04-13 20:21:59 -07:00
|
|
|
#endif
|
|
|
|
|
2021-03-19 13:03:39 -07:00
|
|
|
# define CASESTR(s) case s: return #s
|
|
|
|
|
2006-10-10 01:34:37 +00:00
|
|
|
#ifdef XWFEATURE_RELAY
|
2009-02-07 18:14:28 +00:00
|
|
|
# ifdef DEBUG
|
|
|
|
static const char*
|
|
|
|
relayCmdToStr( XWRELAY_Cmd cmd )
|
|
|
|
{
|
|
|
|
switch( cmd ) {
|
|
|
|
CASESTR( XWRELAY_NONE );
|
|
|
|
CASESTR( XWRELAY_GAME_CONNECT );
|
|
|
|
CASESTR( XWRELAY_GAME_RECONNECT );
|
|
|
|
CASESTR( XWRELAY_GAME_DISCONNECT );
|
|
|
|
CASESTR( XWRELAY_CONNECT_RESP );
|
|
|
|
CASESTR( XWRELAY_RECONNECT_RESP );
|
|
|
|
CASESTR( XWRELAY_ALLHERE );
|
|
|
|
CASESTR( XWRELAY_DISCONNECT_YOU );
|
|
|
|
CASESTR( XWRELAY_DISCONNECT_OTHER );
|
|
|
|
CASESTR( XWRELAY_CONNECTDENIED );
|
2010-09-11 01:23:13 -07:00
|
|
|
#ifdef RELAY_HEARTBEAT
|
2009-02-07 18:14:28 +00:00
|
|
|
CASESTR( XWRELAY_HEARTBEAT );
|
2010-09-11 01:23:13 -07:00
|
|
|
#endif
|
2009-02-07 18:14:28 +00:00
|
|
|
CASESTR( XWRELAY_MSG_FROMRELAY );
|
2011-08-16 19:36:23 -07:00
|
|
|
CASESTR( XWRELAY_MSG_FROMRELAY_NOCONN );
|
2009-02-07 18:14:28 +00:00
|
|
|
CASESTR( XWRELAY_MSG_TORELAY );
|
2011-08-16 19:36:23 -07:00
|
|
|
CASESTR( XWRELAY_MSG_TORELAY_NOCONN );
|
2010-11-11 06:41:24 -08:00
|
|
|
CASESTR( XWRELAY_MSG_STATUS );
|
2014-12-05 06:19:02 -08:00
|
|
|
CASESTR( XWRELAY_ACK );
|
2010-07-30 17:44:20 -07:00
|
|
|
default:
|
2022-04-29 09:19:44 -07:00
|
|
|
XP_LOGFF( "unknown cmd: %d", cmd );
|
2010-07-30 17:44:20 -07:00
|
|
|
XP_ASSERT( 0 );
|
|
|
|
return "<unknown>";
|
2009-02-07 18:14:28 +00:00
|
|
|
}
|
2009-03-06 13:43:03 +00:00
|
|
|
}
|
2009-11-22 18:00:20 +00:00
|
|
|
# endif
|
2009-02-07 18:14:28 +00:00
|
|
|
|
2009-11-08 21:30:13 +00:00
|
|
|
static void
|
2020-04-25 17:47:07 -07:00
|
|
|
got_connect_cmd( CommsCtxt* comms, XWEnv xwe, XWStreamCtxt* stream,
|
2009-12-14 04:03:17 +00:00
|
|
|
XP_Bool reconnected )
|
2009-11-08 21:30:13 +00:00
|
|
|
{
|
2020-08-24 16:16:20 -07:00
|
|
|
LOG_FUNC();
|
2009-11-08 21:30:13 +00:00
|
|
|
XP_U16 nHere, nSought;
|
2010-09-10 01:30:40 -07:00
|
|
|
XP_Bool isServer;
|
2009-11-08 21:30:13 +00:00
|
|
|
|
2020-04-25 17:47:07 -07:00
|
|
|
set_relay_state( comms, xwe, reconnected ? COMMS_RELAYSTATE_RECONNECTED
|
2009-12-14 04:03:17 +00:00
|
|
|
: COMMS_RELAYSTATE_CONNECTED );
|
2014-12-05 06:19:02 -08:00
|
|
|
XWHostID myHostID = stream_getU8( stream );
|
2020-08-24 16:16:20 -07:00
|
|
|
XP_LOGFF( "myHostID: %d", myHostID );
|
2014-12-10 07:54:50 -08:00
|
|
|
if ( comms->rr.myHostID != myHostID ) {
|
2022-04-29 09:19:44 -07:00
|
|
|
XP_LOGFF( "changing rr.myHostID from %x to %x",
|
2020-01-24 09:05:16 -08:00
|
|
|
comms->rr.myHostID, myHostID );
|
2014-12-10 07:54:50 -08:00
|
|
|
comms->rr.myHostID = myHostID;
|
|
|
|
}
|
|
|
|
|
2014-02-07 21:12:45 -08:00
|
|
|
isServer = HOST_ID_SERVER == comms->rr.myHostID;
|
2014-12-05 06:19:02 -08:00
|
|
|
|
2010-09-10 01:30:40 -07:00
|
|
|
if ( isServer != comms->isServer ) {
|
2020-08-24 12:10:25 -07:00
|
|
|
XP_LOGFF( "becoming%s a server", isServer ? "" : " NOT" );
|
2010-09-10 01:30:40 -07:00
|
|
|
comms->isServer = isServer;
|
2020-08-25 14:47:56 -07:00
|
|
|
#ifdef DEBUG
|
|
|
|
XP_U16 queueLen = comms->queueLen;
|
|
|
|
#endif
|
|
|
|
(*comms->rcProc)( xwe, comms->rcClosure, !isServer );
|
|
|
|
XP_ASSERT( queueLen == comms->queueLen ); /* callback should not send!!! */
|
2020-04-25 17:47:07 -07:00
|
|
|
reset_internal( comms, xwe, isServer, comms->rr.nPlayersHere,
|
2014-02-07 21:12:45 -08:00
|
|
|
comms->rr.nPlayersTotal, XP_FALSE );
|
2010-09-10 01:30:40 -07:00
|
|
|
}
|
2009-11-08 21:30:13 +00:00
|
|
|
|
2014-02-07 21:12:45 -08:00
|
|
|
comms->rr.cookieID = stream_getU16( stream );
|
|
|
|
XP_ASSERT( COOKIE_ID_NONE != comms->rr.cookieID );
|
|
|
|
comms->rr.heartbeat = stream_getU16( stream );
|
2009-11-08 21:30:13 +00:00
|
|
|
nSought = (XP_U16)stream_getU8( stream );
|
2009-12-14 04:03:17 +00:00
|
|
|
nHere = (XP_U16)stream_getU8( stream );
|
2010-10-23 20:29:24 -07:00
|
|
|
if ( nSought == nHere ) {
|
2020-04-25 17:47:07 -07:00
|
|
|
set_relay_state( comms, xwe, COMMS_RELAYSTATE_ALLCONNECTED );
|
2010-10-23 20:29:24 -07:00
|
|
|
}
|
2010-09-10 01:30:40 -07:00
|
|
|
|
|
|
|
#ifdef DEBUG
|
|
|
|
{
|
|
|
|
XP_UCHAR connName[MAX_CONNNAME_LEN+1];
|
|
|
|
stringFromStreamHere( stream, connName, sizeof(connName) );
|
2014-02-07 21:12:45 -08:00
|
|
|
if ( comms->rr.connName[0] != '\0'
|
|
|
|
&& 0 != XP_STRCMP( comms->rr.connName, connName ) ) {
|
2022-04-29 09:19:44 -07:00
|
|
|
XP_LOGFF( "we're replacing connNames: %s overwritten by %s",
|
|
|
|
comms->rr.connName, connName );
|
2013-05-30 06:48:29 -07:00
|
|
|
}
|
2014-02-07 21:12:45 -08:00
|
|
|
XP_MEMCPY( comms->rr.connName, connName, sizeof(comms->rr.connName) );
|
2022-04-29 09:19:44 -07:00
|
|
|
XP_LOGFF( "connName: \"%s\" (reconnect=%d)", connName,
|
2013-06-22 05:51:29 -07:00
|
|
|
reconnected );
|
2010-09-10 01:30:40 -07:00
|
|
|
}
|
|
|
|
#else
|
2014-02-07 21:12:45 -08:00
|
|
|
stringFromStreamHere( stream, comms->rr.connName,
|
|
|
|
sizeof(comms->rr.connName) );
|
2010-09-10 01:30:40 -07:00
|
|
|
#endif
|
|
|
|
|
2012-11-03 10:58:01 -07:00
|
|
|
#ifdef XWFEATURE_DEVID
|
2012-11-10 15:05:44 -08:00
|
|
|
DevIDType typ = stream_getU8( stream );
|
|
|
|
XP_UCHAR devID[MAX_DEVID_LEN + 1] = {0};
|
|
|
|
if ( ID_TYPE_NONE != typ ) {
|
|
|
|
stringFromStreamHere( stream, devID, sizeof(devID) );
|
|
|
|
}
|
|
|
|
if ( ID_TYPE_NONE == typ /* error case */
|
|
|
|
|| '\0' != devID[0] ) /* new info case */ {
|
2020-04-26 11:28:18 -07:00
|
|
|
dutil_deviceRegistered( comms->dutil, xwe, typ, devID );
|
2012-11-03 10:58:01 -07:00
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2014-12-05 06:19:02 -08:00
|
|
|
/* Don't bother notifying if the game's already in play on some other
|
|
|
|
transport */
|
|
|
|
if ( CONN_ID_NONE == comms->connID ) {
|
2021-02-07 16:31:32 -08:00
|
|
|
if ( !!comms->procs.rconnd ) {
|
|
|
|
(*comms->procs.rconnd)( xwe, comms->procs.closure,
|
2022-09-03 19:14:40 -07:00
|
|
|
comms->selfAddr.u.ip_relay.invite, reconnected,
|
2021-02-07 16:31:32 -08:00
|
|
|
comms->rr.myHostID, XP_FALSE, nSought - nHere );
|
|
|
|
XP_LOGFF( "have %d of %d players", nHere, nSought );
|
|
|
|
}
|
2014-12-05 06:19:02 -08:00
|
|
|
}
|
2009-11-08 21:30:13 +00:00
|
|
|
setHeartbeatTimer( comms );
|
2010-10-28 18:45:55 -07:00
|
|
|
} /* got_connect_cmd */
|
2009-02-07 18:14:28 +00:00
|
|
|
|
2005-03-19 22:01:38 +00:00
|
|
|
static XP_Bool
|
2020-04-25 17:47:07 -07:00
|
|
|
relayPreProcess( CommsCtxt* comms, XWEnv xwe, XWStreamCtxt* stream, XWHostID* senderID )
|
2005-03-19 22:01:38 +00:00
|
|
|
{
|
2007-02-08 02:53:10 +00:00
|
|
|
XP_Bool consumed = XP_TRUE;
|
2005-03-19 22:01:38 +00:00
|
|
|
XWHostID destID, srcID;
|
2014-02-07 21:12:45 -08:00
|
|
|
CookieID cookieID = comms->rr.cookieID;
|
2011-06-23 07:12:50 -07:00
|
|
|
XWREASON relayErr;
|
2005-03-19 22:01:38 +00:00
|
|
|
|
2005-10-30 16:14:13 +00:00
|
|
|
/* nothing for us to do here if not using relay */
|
2007-02-08 02:53:10 +00:00
|
|
|
XWRELAY_Cmd cmd = stream_getU8( stream );
|
2020-10-23 16:37:28 -07:00
|
|
|
XP_LOGFF( "(%s)", relayCmdToStr( cmd ) );
|
2007-02-08 02:53:10 +00:00
|
|
|
switch( cmd ) {
|
|
|
|
|
|
|
|
case XWRELAY_CONNECT_RESP:
|
2020-04-25 17:47:07 -07:00
|
|
|
got_connect_cmd( comms, xwe, stream, XP_FALSE );
|
2021-03-19 18:51:34 -07:00
|
|
|
send_relay_ack( comms, xwe );
|
2009-11-08 21:30:13 +00:00
|
|
|
break;
|
2007-02-08 02:53:10 +00:00
|
|
|
case XWRELAY_RECONNECT_RESP:
|
2020-04-25 17:47:07 -07:00
|
|
|
got_connect_cmd( comms, xwe, stream, XP_TRUE );
|
|
|
|
comms_resendAll( comms, xwe, COMMS_CONN_NONE, XP_FALSE );
|
2007-02-08 02:53:10 +00:00
|
|
|
break;
|
2005-10-02 15:39:38 +00:00
|
|
|
|
2007-02-08 02:53:10 +00:00
|
|
|
case XWRELAY_ALLHERE:
|
2009-08-21 12:00:09 +00:00
|
|
|
srcID = (XWHostID)stream_getU8( stream );
|
2014-02-07 21:12:45 -08:00
|
|
|
if ( comms->rr.myHostID != HOST_ID_NONE
|
|
|
|
&& comms->rr.myHostID != srcID ) {
|
2022-04-29 09:19:44 -07:00
|
|
|
XP_LOGFF( "changing hostid from %d to %d",
|
2014-02-07 21:12:45 -08:00
|
|
|
comms->rr.myHostID, srcID );
|
2013-06-26 00:42:55 -07:00
|
|
|
}
|
2011-06-27 18:30:52 -07:00
|
|
|
|
2014-03-08 12:47:58 -08:00
|
|
|
if ( COOKIE_ID_NONE == comms->rr.cookieID ) {
|
2022-04-29 09:19:44 -07:00
|
|
|
XP_LOGFF( "cookieID still 0; background send?" );
|
2011-11-21 17:38:50 -08:00
|
|
|
}
|
2011-06-27 18:30:52 -07:00
|
|
|
|
2019-11-27 11:46:31 -08:00
|
|
|
if ( srcID != comms->rr.myHostID ) {
|
2022-04-29 09:19:44 -07:00
|
|
|
XP_LOGFF( "set hostID: %x (was %x)", srcID, comms->rr.myHostID );
|
2019-11-27 11:46:31 -08:00
|
|
|
}
|
2014-12-05 06:19:02 -08:00
|
|
|
comms->rr.myHostID = srcID;
|
2005-09-03 18:31:32 +00:00
|
|
|
|
2009-09-26 14:37:49 +00:00
|
|
|
#ifdef DEBUG
|
2011-11-21 17:38:50 -08:00
|
|
|
{
|
|
|
|
XP_UCHAR connName[MAX_CONNNAME_LEN+1];
|
|
|
|
stringFromStreamHere( stream, connName, sizeof(connName) );
|
2014-02-07 21:12:45 -08:00
|
|
|
if ( comms->rr.connName[0] != '\0'
|
|
|
|
&& 0 != XP_STRCMP( comms->rr.connName, connName ) ) {
|
2022-04-29 09:19:44 -07:00
|
|
|
XP_LOGFF( "we're replacing connNames: %s overwritten by %s",
|
|
|
|
comms->rr.connName, connName );
|
2013-05-30 06:48:29 -07:00
|
|
|
}
|
2014-02-07 21:12:45 -08:00
|
|
|
XP_MEMCPY( comms->rr.connName, connName,
|
|
|
|
sizeof(comms->rr.connName) );
|
2022-04-29 09:19:44 -07:00
|
|
|
XP_LOGFF( "connName: \"%s\"", connName );
|
2011-11-21 17:38:50 -08:00
|
|
|
}
|
2009-09-26 14:37:49 +00:00
|
|
|
#else
|
2014-02-07 21:12:45 -08:00
|
|
|
stringFromStreamHere( stream, comms->rr.connName,
|
|
|
|
sizeof(comms->rr.connName) );
|
2009-09-26 14:37:49 +00:00
|
|
|
#endif
|
|
|
|
|
2011-11-21 17:38:50 -08: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. But don't resend if we were previously
|
|
|
|
REconnected, as we'll have sent then. -- I don't see any send
|
|
|
|
on RECONNECTED, so removing the test for now to fix recon
|
|
|
|
problems on android. */
|
2014-02-07 21:12:45 -08:00
|
|
|
/* if ( COMMS_RELAYSTATE_RECONNECTED != comms->rr.relayState ) { */
|
2020-04-25 17:47:07 -07:00
|
|
|
comms_resendAll( comms, xwe, COMMS_CONN_NONE, XP_FALSE );
|
2011-11-21 17:38:50 -08:00
|
|
|
/* } */
|
|
|
|
if ( XWRELAY_ALLHERE == cmd ) { /* initial connect? */
|
2020-04-25 17:47:07 -07:00
|
|
|
(*comms->procs.rconnd)( xwe, comms->procs.closure,
|
2022-09-03 19:14:40 -07:00
|
|
|
comms->selfAddr.u.ip_relay.invite, XP_FALSE,
|
2014-02-07 21:12:45 -08:00
|
|
|
comms->rr.myHostID, XP_TRUE, 0 );
|
2009-12-14 04:03:17 +00:00
|
|
|
}
|
2020-04-25 17:47:07 -07:00
|
|
|
set_relay_state( comms, xwe, COMMS_RELAYSTATE_ALLCONNECTED );
|
2007-02-08 02:53:10 +00:00
|
|
|
break;
|
|
|
|
case XWRELAY_MSG_FROMRELAY:
|
|
|
|
cookieID = stream_getU16( stream );
|
2011-08-16 19:36:23 -07:00
|
|
|
case XWRELAY_MSG_FROMRELAY_NOCONN:
|
2007-02-08 02:53:10 +00:00
|
|
|
srcID = stream_getU8( stream );
|
|
|
|
destID = stream_getU8( stream );
|
2022-04-29 09:19:44 -07:00
|
|
|
XP_LOGFF( "cookieID: %d; srcID: %x; destID: %x",
|
|
|
|
cookieID, srcID, destID );
|
2007-02-08 02:53:10 +00:00
|
|
|
/* If these values don't check out, drop it */
|
2009-11-08 21:30:13 +00:00
|
|
|
|
2011-01-23 21:45:41 -08:00
|
|
|
/* When a message comes in via proxy (rather than a connection) state
|
|
|
|
may not be as expected. Just commenting these out is probably the
|
|
|
|
wrong fix. Maybe instead the constructor takes a flag that means
|
|
|
|
"assume you're connected" Revisit this. */
|
2014-02-07 21:12:45 -08:00
|
|
|
/* XP_ASSERT( COMMS_RELAYSTATE_ALLCONNECTED == comms->rr.relayState */
|
|
|
|
/* || COMMS_RELAYSTATE_CONNECTED == comms->rr.relayState */
|
|
|
|
/* || COMMS_RELAYSTATE_RECONNECTED == comms->rr.relayState ); */
|
2010-10-12 19:16:46 -07:00
|
|
|
|
2014-02-07 21:12:45 -08:00
|
|
|
if ( destID == comms->rr.myHostID ) { /* When would this not happen? */
|
2010-10-12 19:16:46 -07:00
|
|
|
consumed = XP_FALSE;
|
2014-02-07 21:12:45 -08:00
|
|
|
} else if ( cookieID == comms->rr.cookieID ) {
|
2022-04-29 09:19:44 -07:00
|
|
|
XP_LOGFF( "keeping message though hostID not what "
|
|
|
|
"expected (%d vs %d)", destID, comms->rr.myHostID );
|
2010-10-12 19:16:46 -07:00
|
|
|
consumed = XP_FALSE;
|
2009-11-08 21:30:13 +00:00
|
|
|
}
|
2010-10-12 19:16:46 -07:00
|
|
|
|
2007-02-08 02:53:10 +00:00
|
|
|
if ( consumed ) {
|
2022-04-29 09:19:44 -07:00
|
|
|
XP_LOGFF( "rejecting data message (consumed)" );
|
2007-02-08 02:53:10 +00:00
|
|
|
} else {
|
|
|
|
*senderID = srcID;
|
|
|
|
}
|
|
|
|
break;
|
2005-09-03 06:57:01 +00:00
|
|
|
|
2007-02-08 02:53:10 +00:00
|
|
|
case XWRELAY_DISCONNECT_OTHER:
|
|
|
|
relayErr = stream_getU8( stream );
|
|
|
|
srcID = stream_getU8( stream );
|
2022-04-29 09:19:44 -07:00
|
|
|
XP_LOGFF( "host id %x disconnected", srcID );
|
2009-12-04 08:14:03 +00:00
|
|
|
/* if we don't have connName then RECONNECTED is the wrong state to
|
|
|
|
change to. */
|
2014-02-09 19:35:17 -08:00
|
|
|
if ( COMMS_RELAYSTATE_RECONNECTED < comms->rr.relayState ) {
|
|
|
|
XP_ASSERT( 0 != comms->rr.connName[0] );
|
2016-08-13 17:16:20 -07:00
|
|
|
// XP_ASSERT( COOKIE_ID_NONE != comms->rr.cookieID ); /* firing!! */
|
|
|
|
if ( COOKIE_ID_NONE == comms->rr.cookieID ) { /* firing!! */
|
2022-04-29 09:19:44 -07:00
|
|
|
XP_LOGFF( "cookieID still COOKIE_ID_NONE; dropping!" );
|
2016-08-13 17:16:20 -07:00
|
|
|
} else {
|
2020-04-25 17:47:07 -07:00
|
|
|
set_relay_state( comms, xwe, COMMS_RELAYSTATE_RECONNECTED );
|
2014-02-09 19:35:17 -08:00
|
|
|
/* we will eventually want to tell the user which player's gone */
|
2020-04-26 11:28:18 -07:00
|
|
|
util_userError( comms->util, xwe, ERR_RELAY_BASE + relayErr );
|
2016-08-13 17:16:20 -07:00
|
|
|
}
|
2014-02-09 19:35:17 -08:00
|
|
|
}
|
2007-02-08 02:53:10 +00:00
|
|
|
break;
|
2005-09-03 15:37:49 +00:00
|
|
|
|
2007-02-08 02:53:10 +00:00
|
|
|
case XWRELAY_DISCONNECT_YOU: /* Close socket for this? */
|
|
|
|
relayErr = stream_getU8( stream );
|
2020-04-25 17:47:07 -07:00
|
|
|
set_relay_state( comms, xwe, COMMS_RELAYSTATE_UNCONNECTED );
|
2020-04-26 11:28:18 -07:00
|
|
|
util_userError( comms->util, xwe, ERR_RELAY_BASE + relayErr );
|
2009-12-04 08:14:03 +00:00
|
|
|
break;
|
|
|
|
|
2010-11-11 06:41:24 -08:00
|
|
|
case XWRELAY_MSG_STATUS:
|
|
|
|
relayErr = stream_getU8( stream );
|
2020-04-25 17:47:07 -07:00
|
|
|
(*comms->procs.rerror)( xwe, comms->procs.closure, relayErr );
|
2010-11-11 06:41:24 -08:00
|
|
|
break;
|
|
|
|
|
2009-12-04 08:14:03 +00:00
|
|
|
case XWRELAY_CONNECTDENIED: /* socket will get closed by relay */
|
|
|
|
relayErr = stream_getU8( stream );
|
2022-04-29 09:19:44 -07:00
|
|
|
XP_LOGFF( "got reason: %s", XWREASON2Str( relayErr ) );
|
2020-04-25 17:47:07 -07:00
|
|
|
set_relay_state( comms, xwe, COMMS_RELAYSTATE_DENIED );
|
2011-06-23 07:12:50 -07:00
|
|
|
|
|
|
|
if ( XWRELAY_ERROR_NORECONN == relayErr ) {
|
2020-04-25 17:47:07 -07:00
|
|
|
init_relay( comms, xwe, comms->rr.nPlayersHere, comms->rr.nPlayersTotal );
|
2011-06-23 07:12:50 -07:00
|
|
|
} else {
|
2020-04-26 11:28:18 -07:00
|
|
|
util_userError( comms->util, xwe, ERR_RELAY_BASE + relayErr );
|
2011-06-23 07:12:50 -07:00
|
|
|
/* requires action, not just notification */
|
2020-04-25 17:47:07 -07:00
|
|
|
(*comms->procs.rerror)( xwe, comms->procs.closure, relayErr );
|
2011-06-23 07:12:50 -07:00
|
|
|
}
|
2009-12-04 08:14:03 +00:00
|
|
|
break;
|
|
|
|
|
2007-02-08 02:53:10 +00:00
|
|
|
/* fallthru */
|
|
|
|
default:
|
2014-11-05 07:41:20 -08:00
|
|
|
XP_ASSERT( 0 ); /* while debugging multi-addr, this needs a fix! */
|
2022-04-29 09:19:44 -07:00
|
|
|
XP_LOGFF( "dropping relay msg with cmd %d", (XP_U16)cmd );
|
2007-02-08 02:53:10 +00:00
|
|
|
}
|
2005-03-19 22:01:38 +00:00
|
|
|
|
2020-10-23 16:37:28 -07:00
|
|
|
LOG_RETURNF( "consumed=%s", boolToStr(consumed) );
|
2007-02-08 02:53:10 +00:00
|
|
|
return consumed;
|
|
|
|
} /* relayPreProcess */
|
|
|
|
#endif
|
|
|
|
|
2007-12-05 06:33:37 +00:00
|
|
|
#ifdef COMMS_HEARTBEAT
|
|
|
|
static void
|
|
|
|
noteHBReceived( CommsCtxt* comms/* , const CommsAddrRec* addr */ )
|
|
|
|
{
|
2020-04-26 11:28:18 -07:00
|
|
|
comms->lastMsgRcvdTime = dutil_getCurSeconds( comms->dutil, xwe );
|
2007-12-05 06:33:37 +00:00
|
|
|
setHeartbeatTimer( comms );
|
|
|
|
}
|
|
|
|
#else
|
|
|
|
# define noteHBReceived(a)
|
|
|
|
#endif
|
|
|
|
|
2012-02-07 17:38:24 -08:00
|
|
|
#if defined XWFEATURE_IP_DIRECT
|
2007-02-08 02:53:10 +00:00
|
|
|
static XP_Bool
|
2007-11-26 02:58:25 +00:00
|
|
|
btIpPreProcess( CommsCtxt* comms, XWStreamCtxt* stream )
|
2007-02-08 02:53:10 +00:00
|
|
|
{
|
2007-11-26 02:58:25 +00:00
|
|
|
BTIPMsgType typ = (BTIPMsgType)stream_getU8( stream );
|
|
|
|
XP_Bool consumed = typ != BTIPMSG_DATA;
|
2007-02-08 02:53:10 +00:00
|
|
|
|
|
|
|
if ( consumed ) {
|
2007-11-26 02:58:25 +00:00
|
|
|
/* This is all there is so far */
|
2007-12-05 06:33:37 +00:00
|
|
|
if ( typ == BTIPMSG_RESET ) {
|
2012-11-13 21:25:03 -08:00
|
|
|
(void)comms_resendAll( comms, XP_FALSE );
|
2007-12-05 06:33:37 +00:00
|
|
|
} else if ( typ == BTIPMSG_HB ) {
|
|
|
|
/* noteHBReceived( comms, addr ); */
|
|
|
|
} else {
|
|
|
|
XP_ASSERT( 0 );
|
|
|
|
}
|
2005-03-19 22:01:38 +00:00
|
|
|
}
|
2007-02-08 02:53:10 +00:00
|
|
|
|
2005-03-19 22:01:38 +00:00
|
|
|
return consumed;
|
2007-11-26 02:58:25 +00:00
|
|
|
} /* btIpPreProcess */
|
2005-07-23 15:28:15 +00:00
|
|
|
#endif
|
2005-03-19 22:01:38 +00:00
|
|
|
|
2007-11-26 02:58:25 +00:00
|
|
|
static XP_Bool
|
2022-05-14 07:10:48 -07:00
|
|
|
preProcess(
|
|
|
|
#ifdef XWFEATURE_RELAY
|
|
|
|
CommsCtxt* comms, XWEnv xwe, XWStreamCtxt* stream,
|
|
|
|
XP_Bool* usingRelay, XWHostID* senderID,
|
|
|
|
#endif
|
|
|
|
const CommsAddrRec* useAddr
|
|
|
|
)
|
2007-11-26 02:58:25 +00:00
|
|
|
{
|
|
|
|
XP_Bool consumed = XP_FALSE;
|
2014-11-05 07:41:20 -08:00
|
|
|
|
2014-11-11 07:46:01 -08:00
|
|
|
/* There should be exactly one type associated with an incoming message */
|
|
|
|
CommsConnType typ = addr_getType( useAddr );
|
2020-10-23 16:37:28 -07:00
|
|
|
XP_LOGFF( "(typ=%s)", ConnType2Str(typ) );
|
2014-11-05 07:41:20 -08:00
|
|
|
|
2014-11-11 07:46:01 -08:00
|
|
|
switch ( typ ) {
|
2007-11-26 02:58:25 +00:00
|
|
|
#ifdef XWFEATURE_RELAY
|
2014-11-11 07:46:01 -08:00
|
|
|
/* 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:
|
2020-04-25 17:47:07 -07:00
|
|
|
consumed = relayPreProcess( comms, xwe, stream, senderID );
|
2014-11-11 07:46:01 -08:00
|
|
|
if ( !consumed ) {
|
|
|
|
*usingRelay = XP_TRUE;
|
|
|
|
}
|
|
|
|
break;
|
2007-11-26 02:58:25 +00:00
|
|
|
#endif
|
2012-02-07 17:38:24 -08:00
|
|
|
#if defined XWFEATURE_IP_DIRECT
|
2014-11-11 07:46:01 -08:00
|
|
|
case COMMS_CONN_BT:
|
|
|
|
case COMMS_CONN_IP_DIRECT:
|
|
|
|
consumed = btIpPreProcess( comms, stream );
|
|
|
|
break;
|
2014-11-05 07:41:20 -08:00
|
|
|
#endif
|
|
|
|
#if defined XWFEATURE_SMS
|
2014-11-11 07:46:01 -08:00
|
|
|
case COMMS_CONN_SMS:
|
|
|
|
break; /* nothing to grab */
|
2014-11-14 07:06:02 -08:00
|
|
|
#endif
|
|
|
|
#ifdef XWFEATURE_BLUETOOTH
|
|
|
|
case COMMS_CONN_BT:
|
|
|
|
break; /* nothing to grab */
|
2007-11-26 02:58:25 +00:00
|
|
|
#endif
|
2016-11-14 08:06:53 -08:00
|
|
|
case COMMS_CONN_P2P:
|
|
|
|
break; /* nothing to grab?? */
|
2019-11-23 10:53:51 -08:00
|
|
|
case COMMS_CONN_NFC:
|
2020-05-20 13:58:53 -07:00
|
|
|
case COMMS_CONN_MQTT:
|
2019-11-23 10:53:51 -08:00
|
|
|
break; /* nothing to grab?? */
|
2014-11-11 07:46:01 -08:00
|
|
|
default:
|
|
|
|
XP_ASSERT(0);
|
|
|
|
break;
|
2007-11-26 02:58:25 +00:00
|
|
|
}
|
2017-01-22 14:15:29 -08:00
|
|
|
LOG_RETURNF( "%s", boolToStr(consumed) );
|
2007-11-26 02:58:25 +00:00
|
|
|
return consumed;
|
|
|
|
} /* preProcess */
|
|
|
|
|
2007-12-05 06:33:37 +00:00
|
|
|
static AddressRecord*
|
2023-02-15 08:31:51 -08:00
|
|
|
getRecordFor( const CommsCtxt* comms, const XP_PlayerAddr channelNo )
|
2006-11-11 22:37:36 +00:00
|
|
|
{
|
2007-12-05 06:33:37 +00:00
|
|
|
AddressRecord* rec;
|
2007-12-31 20:00:13 +00:00
|
|
|
|
2008-01-07 01:10:43 +00:00
|
|
|
/* Use addr if we have it. Otherwise use channelNo if non-0 */
|
2014-11-06 20:28:51 -08:00
|
|
|
CNO_FMT( cbuf, channelNo );
|
2007-12-05 06:33:37 +00:00
|
|
|
for ( rec = comms->recs; !!rec; rec = rec->next ) {
|
2022-12-26 19:24:39 -08:00
|
|
|
/* guest should have only one rec max */
|
|
|
|
XP_ASSERT( comms->isServer || !rec->next );
|
2014-11-11 07:46:01 -08:00
|
|
|
|
2014-11-06 20:28:51 -08:00
|
|
|
CNO_FMT( cbuf1, rec->channelNo );
|
2022-04-29 09:19:44 -07:00
|
|
|
XP_LOGFF( "comparing rec channel %s with addr channel %s",
|
2014-11-06 20:28:51 -08:00
|
|
|
cbuf1, cbuf );
|
2014-11-05 07:41:20 -08:00
|
|
|
|
2023-02-16 21:36:46 -08:00
|
|
|
/* Invite case: base on channelNo bits if the rest is 0 */
|
|
|
|
if ( (0 == (rec->channelNo & ~CHANNEL_MASK)) && (0 == (channelNo & ~CHANNEL_MASK)) ) {
|
|
|
|
if ( rec->channelNo == channelNo ) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
} else if ( (rec->channelNo & ~CHANNEL_MASK) == (channelNo & ~CHANNEL_MASK) ) {
|
|
|
|
XP_LOGFF( "match based on channels!!!" );
|
|
|
|
/* This is so wrong for addresses coming from invites. Why works
|
|
|
|
with GTK? */
|
2007-12-05 06:33:37 +00:00
|
|
|
break;
|
2006-11-11 22:37:36 +00:00
|
|
|
}
|
|
|
|
}
|
2014-11-06 20:28:51 -08:00
|
|
|
|
2022-10-07 09:23:09 -07:00
|
|
|
XP_LOGFF( "(%s) => %p", cbuf, rec );
|
2007-12-05 06:33:37 +00:00
|
|
|
return rec;
|
2009-01-05 02:07:30 +00:00
|
|
|
} /* getRecordFor */
|
2007-12-05 06:33:37 +00:00
|
|
|
|
2022-10-08 12:26:00 -07:00
|
|
|
/* It should be possible to find the next channel from what's already in use,
|
|
|
|
and so to get rid of nextChannelNo, but this doesn't work yet. Saving the
|
|
|
|
code for later... */
|
|
|
|
static XP_PlayerAddr
|
|
|
|
getNextChannelNo( CommsCtxt* comms )
|
|
|
|
{
|
|
|
|
XP_PlayerAddr result = ++comms->nextChannelNo;
|
|
|
|
#if 0
|
|
|
|
XP_U16 mask = 0;
|
|
|
|
for ( AddressRecord* rec = comms->recs; !!rec; rec = rec->next ) {
|
|
|
|
XP_U16 forceChannel = rec->channelNo & CHANNEL_MASK;
|
|
|
|
XP_ASSERT( forceChannel <= CHANNEL_MASK );
|
|
|
|
XP_LOGFF( "forceChannel: %d", forceChannel );
|
|
|
|
mask |= 1 << forceChannel;
|
|
|
|
XP_LOGFF( "mask now: %x", mask );
|
|
|
|
}
|
|
|
|
|
|
|
|
XP_PlayerAddr candidate;
|
|
|
|
for ( candidate = 1; ; ++candidate ) {
|
|
|
|
XP_ASSERT( candidate <= CHANNEL_MASK );
|
|
|
|
if ( 0 == ((1 << candidate) & mask) ) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if ( candidate != result ) {
|
|
|
|
XP_LOGFF( "ERROR: candidate: %d; result: %d", candidate, result );
|
|
|
|
}
|
|
|
|
// XP_ASSERT( candidate == result );
|
|
|
|
#endif
|
|
|
|
LOG_RETURNF( "%d", result );
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
2014-12-21 17:40:00 -08:00
|
|
|
static XP_Bool
|
|
|
|
checkChannelNo( CommsCtxt* comms, XP_PlayerAddr* channelNoP )
|
|
|
|
{
|
2017-01-22 14:15:29 -08:00
|
|
|
XP_Bool success = XP_TRUE;
|
2014-12-21 17:40:00 -08:00
|
|
|
XP_PlayerAddr channelNo = *channelNoP;
|
|
|
|
if ( 0 == (channelNo & CHANNEL_MASK) ) {
|
2017-06-28 07:04:15 -07:00
|
|
|
success = comms->nextChannelNo < CHANNEL_MASK;
|
|
|
|
if ( success ) {
|
2022-10-08 12:26:00 -07:00
|
|
|
channelNo |= getNextChannelNo( comms );
|
2022-10-08 10:41:29 -07:00
|
|
|
CNO_FMT( cbuf, channelNo );
|
|
|
|
XP_LOGFF( "assigned channelNo: %s", cbuf );
|
2017-06-28 07:04:15 -07:00
|
|
|
}
|
|
|
|
// XP_ASSERT( comms->nextChannelNo <= CHANNEL_MASK );
|
2014-12-21 17:40:00 -08:00
|
|
|
} else {
|
|
|
|
/* Let's make sure we don't assign it later */
|
2022-10-08 12:26:00 -07:00
|
|
|
XP_ASSERT( 0 ); /* does this happen */
|
2014-12-21 17:40:00 -08:00
|
|
|
comms->nextChannelNo = channelNo;
|
|
|
|
}
|
|
|
|
*channelNoP = channelNo;
|
2017-01-22 14:15:29 -08:00
|
|
|
LOG_RETURNF( "%s", boolToStr(success) );
|
|
|
|
return success;
|
2014-12-21 17:40:00 -08:00
|
|
|
}
|
|
|
|
|
2007-12-05 06:33:37 +00:00
|
|
|
/* 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 05:35:29 +00:00
|
|
|
*
|
2007-12-05 06:33:37 +00: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 05:35:29 +00:00
|
|
|
*
|
2007-12-05 06:33:37 +00: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 05:35:29 +00:00
|
|
|
*/
|
2007-12-05 06:33:37 +00:00
|
|
|
static AddressRecord*
|
2023-02-15 08:31:51 -08:00
|
|
|
validateInitialMessage( CommsCtxt* comms, XP_Bool XP_UNUSED_HEARTBEAT(hasPayload),
|
2007-12-05 06:33:37 +00:00
|
|
|
const CommsAddrRec* addr, XWHostID senderID,
|
2022-10-08 12:26:00 -07:00
|
|
|
XP_PlayerAddr* channelNoP, XP_U16 flags, MsgID msgID )
|
2007-12-05 06:33:37 +00:00
|
|
|
{
|
2022-10-08 12:26:00 -07:00
|
|
|
CNO_FMT( cbuf, *channelNoP );
|
2021-12-15 10:03:09 -08:00
|
|
|
XP_LOGFF( TAGFMT(%s), TAGPRMS, cbuf );
|
2014-11-06 20:28:51 -08:00
|
|
|
|
2009-02-07 18:14:28 +00:00
|
|
|
AddressRecord* rec = NULL;
|
|
|
|
if ( 0 ) {
|
|
|
|
#ifdef COMMS_HEARTBEAT
|
|
|
|
} else if ( comms->doHeartbeat ) {
|
|
|
|
XP_Bool addRec = XP_FALSE;
|
2009-09-26 14:37:49 +00:00
|
|
|
/* This (with mask) is untested!!! */
|
2023-02-15 08:31:51 -08:00
|
|
|
rec = getRecordFor( comms, *channelNoP );
|
2007-12-05 06:33:37 +00:00
|
|
|
|
2009-02-07 18:14:28 +00:00
|
|
|
if ( hasPayload ) {
|
|
|
|
if ( rec ) {
|
|
|
|
if ( rec->initialSeen ) {
|
|
|
|
rec = NULL; /* reject it! */
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
addRec = XP_TRUE;
|
2007-12-05 06:33:37 +00:00
|
|
|
}
|
|
|
|
} else {
|
2009-02-07 18:14:28 +00:00
|
|
|
/* This is a heartbeat */
|
|
|
|
if ( !rec && comms->isServer ) {
|
|
|
|
addRec = XP_TRUE;
|
|
|
|
}
|
2007-12-05 06:33:37 +00:00
|
|
|
}
|
|
|
|
|
2009-02-07 18:14:28 +00:00
|
|
|
if ( addRec ) {
|
|
|
|
if ( comms->isServer ) {
|
2022-10-08 12:26:00 -07:00
|
|
|
CNO_FMT( cbuf, *channelNoP );
|
2021-12-15 10:03:09 -08:00
|
|
|
XP_LOGFF( TAGFMT() "looking at %s", TAGPRMS, cbuf );
|
2022-10-08 12:26:00 -07:00
|
|
|
XP_ASSERT( (*channelNoP & CHANNEL_MASK) == 0 );
|
|
|
|
*channelNoP |= getNextChannelNo( comms );
|
2014-11-06 20:28:51 -08:00
|
|
|
CNO_FMT( cbuf1, *channelNo );
|
2022-09-06 12:16:29 -07:00
|
|
|
XP_LOGFF( TAGFMT() "ORd channel onto channelNo: now %s", TAGPRMS,
|
|
|
|
cbuf1 );
|
|
|
|
XP_ASSERT( comms->nextChannelNo <= CHANNEL_ID_MASK );
|
2009-02-07 18:14:28 +00:00
|
|
|
}
|
2023-02-15 08:31:51 -08:00
|
|
|
rec = rememberChannelAddress( comms, *channelNo, senderID, addr,
|
2022-09-03 18:47:04 -07:00
|
|
|
flags );
|
2009-02-07 18:14:28 +00:00
|
|
|
if ( hasPayload ) {
|
|
|
|
rec->initialSeen = XP_TRUE;
|
|
|
|
} else {
|
|
|
|
rec = NULL;
|
|
|
|
}
|
2007-12-05 06:33:37 +00:00
|
|
|
}
|
2022-12-27 16:03:08 -08:00
|
|
|
#endif /* ifdef COMMS_HEARTBEAT */
|
2009-02-07 18:14:28 +00:00
|
|
|
} else {
|
2022-10-08 12:26:00 -07:00
|
|
|
CNO_FMT( cbuf, *channelNoP );
|
2021-12-15 10:03:09 -08:00
|
|
|
XP_LOGFF( TAGFMT() "looking at %s", TAGPRMS, cbuf );
|
2023-02-15 08:31:51 -08:00
|
|
|
rec = getRecordFor( comms, *channelNoP );
|
2009-02-07 18:14:28 +00:00
|
|
|
if ( !!rec ) {
|
2020-09-10 15:11:22 -07:00
|
|
|
augmentChannelAddr( comms, rec, addr, senderID );
|
2022-09-03 18:47:04 -07:00
|
|
|
|
|
|
|
/* Used to be that the initial message was where the channel
|
|
|
|
record got created, but now the client creates an address for
|
|
|
|
the host on startup (comms_make()) */
|
2022-10-01 08:43:10 -07:00
|
|
|
if ( comms->isServer || 1 != msgID ) {
|
2022-09-03 18:47:04 -07:00
|
|
|
XP_LOGFF( TAGFMT() "rejecting duplicate INIT message", TAGPRMS );
|
|
|
|
rec = NULL;
|
|
|
|
} else {
|
|
|
|
XP_LOGFF( "accepting duplicate (?) msg" );
|
|
|
|
}
|
2007-12-05 06:33:37 +00:00
|
|
|
} else {
|
2009-02-07 18:14:28 +00:00
|
|
|
if ( comms->isServer ) {
|
2022-10-08 12:26:00 -07:00
|
|
|
if ( checkChannelNo( comms, channelNoP ) ) {
|
|
|
|
CNO_FMT( cbuf, *channelNoP );
|
2021-12-15 10:03:09 -08:00
|
|
|
XP_LOGFF( TAGFMT() "augmented channel: %s", TAGPRMS, cbuf );
|
2013-07-02 18:49:48 -07:00
|
|
|
} else {
|
|
|
|
/* Why do I sometimes see these in the middle of a game
|
|
|
|
with lots of messages already sent? connID of 0 should
|
|
|
|
only happen at the start! */
|
2021-12-15 10:03:09 -08:00
|
|
|
XP_LOGFF( TAGFMT() "dropping msg because channel already set",
|
2015-02-09 19:15:43 -08:00
|
|
|
TAGPRMS );
|
2013-07-02 18:49:48 -07:00
|
|
|
goto errExit;
|
|
|
|
}
|
2009-02-07 18:14:28 +00:00
|
|
|
}
|
2023-02-15 08:31:51 -08:00
|
|
|
rec = rememberChannelAddress( comms, *channelNoP, senderID,
|
2022-09-06 12:16:29 -07:00
|
|
|
addr, flags );
|
2007-12-05 06:33:37 +00:00
|
|
|
}
|
|
|
|
}
|
2013-07-02 18:49:48 -07:00
|
|
|
errExit:
|
2008-03-11 12:24:23 +00:00
|
|
|
LOG_RETURNF( XP_P, rec );
|
2007-12-05 06:33:37 +00:00
|
|
|
return rec;
|
|
|
|
} /* validateInitialMessage */
|
|
|
|
|
2015-02-18 19:19:36 -08:00
|
|
|
static XP_U16
|
2021-12-28 18:49:56 -08:00
|
|
|
makeFlags( const CommsCtxt* comms, XP_U16 headerLen, MsgID msgID )
|
2015-02-18 19:19:36 -08:00
|
|
|
{
|
|
|
|
XP_U16 flags = COMMS_VERSION;
|
2015-02-23 18:28:31 -08:00
|
|
|
if ( comms->isServer ) {
|
|
|
|
flags |= IS_SERVER_BIT;
|
|
|
|
}
|
2021-12-21 16:23:24 -08:00
|
|
|
if ( CONN_ID_NONE == comms->connID ) {
|
|
|
|
flags |= NO_CONNID_BIT;
|
|
|
|
}
|
2021-12-28 18:49:56 -08:00
|
|
|
if ( 0 == msgID ) {
|
|
|
|
flags |= NO_MSGID_BIT;
|
|
|
|
}
|
2021-12-21 16:23:24 -08:00
|
|
|
|
|
|
|
XP_ASSERT( headerLen == ((headerLen << HEADER_LEN_OFFSET) >> HEADER_LEN_OFFSET) );
|
|
|
|
flags |= headerLen << HEADER_LEN_OFFSET;
|
|
|
|
|
2015-02-18 19:19:36 -08:00
|
|
|
return flags;
|
|
|
|
}
|
|
|
|
|
2007-12-05 06:33:37 +00:00
|
|
|
/* 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*
|
2020-04-25 17:47:07 -07:00
|
|
|
validateChannelMessage( CommsCtxt* comms, XWEnv xwe, const CommsAddrRec* addr,
|
2014-11-11 07:46:01 -08:00
|
|
|
XP_PlayerAddr channelNo, XWHostID senderID,
|
|
|
|
MsgID msgID, MsgID lastMsgRcd )
|
2007-12-05 06:33:37 +00:00
|
|
|
|
|
|
|
{
|
|
|
|
AddressRecord* rec;
|
2023-01-06 07:52:41 -08:00
|
|
|
THREAD_CHECK_START(comms);
|
2007-12-05 06:33:37 +00:00
|
|
|
LOG_FUNC();
|
|
|
|
|
2023-02-15 08:31:51 -08:00
|
|
|
rec = getRecordFor( comms, channelNo );
|
2007-12-05 06:33:37 +00:00
|
|
|
if ( !!rec ) {
|
2020-04-25 17:47:07 -07:00
|
|
|
removeFromQueue( comms, xwe, channelNo, lastMsgRcd );
|
2014-11-11 07:46:01 -08:00
|
|
|
|
2020-09-10 15:11:22 -07:00
|
|
|
augmentChannelAddr( comms, rec, addr, senderID );
|
2014-11-11 07:46:01 -08:00
|
|
|
|
2019-02-04 13:21:24 -08:00
|
|
|
if ( msgID == rec->lastMsgRcd + 1 ) {
|
2021-12-15 10:03:09 -08:00
|
|
|
XP_LOGFF( TAGFMT() "expected %d AND got %d", TAGPRMS,
|
2019-02-04 13:21:24 -08:00
|
|
|
msgID, msgID );
|
|
|
|
} else if ( msgID != rec->lastMsgRcd + 1 ) {
|
2021-12-15 10:03:09 -08:00
|
|
|
XP_LOGFF( TAGFMT() "expected %d, got %d", TAGPRMS,
|
2007-12-05 06:33:37 +00:00
|
|
|
rec->lastMsgRcd + 1, msgID );
|
2023-11-17 19:39:45 -08:00
|
|
|
ackAnyImpl( comms, xwe, XP_TRUE );
|
2007-12-05 06:33:37 +00:00
|
|
|
rec = NULL;
|
|
|
|
}
|
|
|
|
} else {
|
2014-11-06 18:44:19 -08:00
|
|
|
CNO_FMT( cbuf, channelNo );
|
2021-12-15 10:03:09 -08:00
|
|
|
XP_LOGFF( TAGFMT() "no rec for %s", TAGPRMS, cbuf );
|
2007-12-05 06:33:37 +00:00
|
|
|
}
|
|
|
|
|
2008-03-11 12:24:23 +00:00
|
|
|
LOG_RETURNF( XP_P, rec );
|
2023-01-06 07:52:41 -08:00
|
|
|
THREAD_CHECK_END();
|
2007-12-05 06:33:37 +00:00
|
|
|
return rec;
|
|
|
|
} /* validateChannelMessage */
|
|
|
|
|
2021-12-28 18:49:56 -08:00
|
|
|
typedef struct _HeaderStuff {
|
|
|
|
XP_U16 flags;
|
|
|
|
XP_U32 connID;
|
|
|
|
XP_PlayerAddr channelNo;
|
|
|
|
MsgID msgID;
|
|
|
|
MsgID lastMsgRcd;
|
|
|
|
} HeaderStuff;
|
|
|
|
|
|
|
|
static XP_Bool
|
|
|
|
getCheckChannelSeed( CommsCtxt* comms, XWStreamCtxt* stream, HeaderStuff* stuff )
|
|
|
|
{
|
2023-01-06 07:52:41 -08:00
|
|
|
XP_Bool messageValid;
|
|
|
|
THREAD_CHECK_START(comms);
|
|
|
|
messageValid = stream_gotU16( stream, &stuff->channelNo );
|
2021-12-28 18:49:56 -08:00
|
|
|
if ( messageValid ) {
|
|
|
|
XP_U16 channelSeed = comms_getChannelSeed( comms );
|
|
|
|
XP_U16 flags = stuff->flags;
|
|
|
|
|
|
|
|
/* First test isn't valid if we haven't passed the bit explicitly */
|
|
|
|
if ( 0 != flags && (comms->isServer == (0 != (flags & IS_SERVER_BIT))) ) {
|
|
|
|
XP_LOGFF( TAGFMT() "server bits mismatch; isServer: %d; flags: %x",
|
|
|
|
TAGPRMS, comms->isServer, flags );
|
|
|
|
messageValid = XP_FALSE;
|
|
|
|
} else if ( comms->isServer ) {
|
|
|
|
/* channelNo comparison invalid */
|
|
|
|
} else if ( 0 == stuff->channelNo || 0 == channelSeed ) {
|
|
|
|
XP_LOGFF( TAGFMT() "one of channelNos still 0", TAGPRMS );
|
|
|
|
XP_ASSERT(0);
|
2022-09-06 12:16:29 -07:00
|
|
|
} else if ( (stuff->channelNo & ~CHANNEL_MASK)
|
|
|
|
!= (channelSeed & ~CHANNEL_MASK) ) {
|
|
|
|
XP_LOGFF( "channelNos test fails: %x vs %x", stuff->channelNo,
|
|
|
|
channelSeed );
|
2021-12-28 18:49:56 -08:00
|
|
|
messageValid = XP_FALSE;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
LOG_RETURNF( "%s", boolToStr(messageValid) );
|
2023-01-06 07:52:41 -08:00
|
|
|
THREAD_CHECK_END();
|
2021-12-28 18:49:56 -08:00
|
|
|
return messageValid;
|
|
|
|
}
|
|
|
|
|
|
|
|
static XP_Bool
|
|
|
|
parseBeefHeader( CommsCtxt* comms, XWStreamCtxt* stream, HeaderStuff* stuff )
|
|
|
|
{
|
|
|
|
XP_Bool messageValid =
|
|
|
|
stream_gotU16( stream, &stuff->flags ) /* flags are the next short */
|
|
|
|
&& stream_gotU32( stream, &stuff->connID );
|
|
|
|
XP_LOGFF( TAGFMT() "read connID (gameID) of %x", TAGPRMS, stuff->connID );
|
|
|
|
|
|
|
|
messageValid = messageValid
|
|
|
|
&& getCheckChannelSeed( comms, stream, stuff )
|
|
|
|
&& stream_gotU32( stream, &stuff->msgID )
|
|
|
|
&& stream_gotU32( stream, &stuff->lastMsgRcd );
|
|
|
|
|
|
|
|
LOG_RETURNF( "%s", boolToStr(messageValid) );
|
|
|
|
return messageValid;
|
|
|
|
}
|
|
|
|
|
|
|
|
static XP_Bool
|
2023-02-11 07:48:31 -08:00
|
|
|
parseSmallHeader( CommsCtxt* comms, XWStreamCtxt* msgStream,
|
2021-12-28 18:49:56 -08:00
|
|
|
HeaderStuff* stuff )
|
|
|
|
{
|
|
|
|
XP_Bool messageValid = XP_FALSE;
|
2023-01-06 07:52:41 -08:00
|
|
|
THREAD_CHECK_START(comms);
|
2021-12-28 18:49:56 -08:00
|
|
|
XP_U16 headerLen = stuff->flags >> HEADER_LEN_OFFSET;
|
|
|
|
XP_ASSERT( 0 < headerLen );
|
|
|
|
XP_ASSERT( headerLen <= stream_getSize( msgStream ) );
|
|
|
|
if ( headerLen <= stream_getSize( msgStream ) ) {
|
2022-09-06 12:16:29 -07:00
|
|
|
XWStreamCtxt* hdrStream =
|
|
|
|
mem_stream_make_raw( MPPARM(comms->mpool)
|
|
|
|
dutil_getVTManager(comms->dutil));
|
2021-12-28 18:49:56 -08:00
|
|
|
stream_getFromStream( hdrStream, msgStream, headerLen );
|
|
|
|
stuff->connID = 0 == (stuff->flags & NO_CONNID_BIT)
|
|
|
|
? comms->util->gameInfo->gameID : CONN_ID_NONE;
|
|
|
|
|
|
|
|
if ( getCheckChannelSeed( comms, hdrStream, stuff ) ) {
|
|
|
|
XP_ASSERT( stuff->msgID == 0 );
|
|
|
|
if ( 0 == (stuff->flags & NO_MSGID_BIT) ) {
|
|
|
|
stuff->msgID = stream_getU32VL( hdrStream );
|
|
|
|
}
|
|
|
|
stuff->lastMsgRcd = stream_getU32VL( hdrStream );
|
|
|
|
messageValid = XP_TRUE;
|
|
|
|
}
|
2023-02-03 22:00:47 -08:00
|
|
|
stream_destroy( hdrStream );
|
2021-12-28 18:49:56 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
LOG_RETURNF( "%s", boolToStr(messageValid) );
|
2023-01-06 07:52:41 -08:00
|
|
|
THREAD_CHECK_END();
|
2021-12-28 18:49:56 -08:00
|
|
|
return messageValid;
|
|
|
|
}
|
|
|
|
|
2003-11-01 05:35:29 +00:00
|
|
|
XP_Bool
|
2020-04-25 17:47:07 -07:00
|
|
|
comms_checkIncomingStream( CommsCtxt* comms, XWEnv xwe, XWStreamCtxt* stream,
|
2015-02-07 13:24:09 -08:00
|
|
|
const CommsAddrRec* retAddr, CommsMsgState* state )
|
2003-11-01 05:35:29 +00:00
|
|
|
{
|
2023-01-06 07:52:41 -08:00
|
|
|
XP_Bool messageValid = XP_FALSE;
|
|
|
|
THREAD_CHECK_START(comms);
|
2014-11-11 07:46:01 -08:00
|
|
|
XP_ASSERT( !!retAddr ); /* for now */
|
2015-02-08 09:58:34 -08:00
|
|
|
XP_MEMSET( state, 0, sizeof(*state) );
|
2015-02-08 10:37:54 -08:00
|
|
|
#ifdef DEBUG
|
|
|
|
state->comms = comms;
|
2022-09-09 11:23:30 -07:00
|
|
|
if ( comms->processingMsg ) {
|
|
|
|
XP_LOGFF( "processingMsg SET, so dropping message" );
|
|
|
|
return XP_FALSE;
|
|
|
|
}
|
2015-02-08 10:37:54 -08:00
|
|
|
XP_ASSERT( !comms->processingMsg );
|
|
|
|
comms->processingMsg = XP_TRUE;
|
2019-11-27 08:43:40 -08:00
|
|
|
CommsConnType addrType = addr_getType( retAddr );
|
2015-02-08 10:37:54 -08:00
|
|
|
#endif
|
|
|
|
|
2021-12-15 10:03:09 -08:00
|
|
|
XP_LOGFF( TAGFMT(retAddr.typ=%s), TAGPRMS, ConnType2Str(addrType ) );
|
2019-11-23 10:53:51 -08:00
|
|
|
if ( comms_getAddrDisabled( comms, addrType, XP_FALSE ) ) {
|
2022-04-29 09:19:44 -07:00
|
|
|
XP_LOGFF( "dropping message because %s disabled",
|
2019-11-23 10:53:51 -08:00
|
|
|
ConnType2Str( addrType ) );
|
2022-09-03 19:14:40 -07:00
|
|
|
/* } else if (0 == (comms->selfAddr._conTypes & retAddr->_conTypes)) { */
|
2020-02-24 21:10:49 +01:00
|
|
|
/* /\* we don't expect messages with that address type; drop it *\/ */
|
|
|
|
/* XP_LOGF( "%s: not expecting %s messages", __func__, */
|
|
|
|
/* ConnType2Str( addrType ) ); */
|
2014-11-06 06:35:28 -08:00
|
|
|
} else {
|
2020-03-28 16:25:30 -07:00
|
|
|
#ifdef DEBUG
|
2022-09-03 19:14:40 -07:00
|
|
|
if (0 == (comms->selfAddr._conTypes & retAddr->_conTypes)) {
|
2020-03-28 16:25:30 -07:00
|
|
|
XP_LOGFF( "not expecting %s messages (but proceeding)",
|
|
|
|
ConnType2Str( addrType ) );
|
|
|
|
}
|
|
|
|
#endif
|
2014-11-06 06:35:28 -08:00
|
|
|
XWHostID senderID = 0; /* unset; default for non-relay cases */
|
2022-05-14 07:10:48 -07:00
|
|
|
#ifdef XWFEATURE_RELAY
|
2014-11-06 06:35:28 -08:00
|
|
|
XP_Bool usingRelay = XP_FALSE;
|
2022-05-14 07:10:48 -07:00
|
|
|
#endif
|
2005-03-06 17:56:34 +00:00
|
|
|
|
2013-06-30 08:33:29 -07:00
|
|
|
#ifdef COMMS_CHECKSUM
|
2014-11-06 06:35:28 -08:00
|
|
|
XP_U16 initialLen = stream_getSize( stream );
|
2013-06-30 08:33:29 -07:00
|
|
|
#endif
|
2005-03-19 22:01:38 +00:00
|
|
|
|
2022-05-14 07:10:48 -07:00
|
|
|
if ( !preProcess(
|
|
|
|
#ifdef XWFEATURE_RELAY
|
|
|
|
comms, xwe , stream, &usingRelay, &senderID,
|
|
|
|
#endif
|
2023-01-17 18:01:30 -08:00
|
|
|
retAddr ) ) {
|
2013-06-30 08:33:29 -07:00
|
|
|
#ifdef COMMS_CHECKSUM
|
2021-03-15 16:58:17 -07:00
|
|
|
state->len = stream_getSize( stream );
|
|
|
|
// stream_getPtr pts at base, but sum excludes relay header
|
|
|
|
const XP_U8* ptr = initialLen - state->len + stream_getPtr( stream );
|
|
|
|
XP_UCHAR* tmpsum = dutil_md5sum( comms->dutil, xwe, ptr, state->len );
|
|
|
|
XP_STRCAT( state->sum, tmpsum );
|
2021-12-15 10:03:09 -08:00
|
|
|
XP_LOGFF( TAGFMT() "got message of len %d with sum %s",
|
|
|
|
TAGPRMS, state->len, state->sum );
|
2021-03-15 16:58:17 -07:00
|
|
|
XP_FREE( comms->mpool, tmpsum );
|
2013-06-30 08:33:29 -07:00
|
|
|
#endif
|
2021-12-28 18:49:56 -08:00
|
|
|
HeaderStuff stuff = {0};
|
|
|
|
messageValid = stream_gotU16( stream, &stuff.flags );
|
2021-12-21 16:23:24 -08:00
|
|
|
|
2015-02-08 09:58:34 -08:00
|
|
|
if ( messageValid ) {
|
2021-12-21 16:23:24 -08:00
|
|
|
/* If BEEF is next sender is using old format. Otherwise
|
|
|
|
assume the bits are flags and BEEF is skipped by newer
|
|
|
|
code. Should work with anything newer then six years
|
|
|
|
ago. */
|
2021-12-28 18:49:56 -08:00
|
|
|
if ( HAS_VERSION_FLAG == stuff.flags ) {
|
|
|
|
messageValid = parseBeefHeader( comms, stream, &stuff );
|
2021-12-29 07:35:43 -08:00
|
|
|
} else if ( COMMS_VERSION == (stuff.flags & VERSION_MASK) ) {
|
2023-02-11 07:48:31 -08:00
|
|
|
messageValid = parseSmallHeader( comms, stream, &stuff );
|
2021-12-21 16:23:24 -08:00
|
|
|
}
|
|
|
|
}
|
2022-09-17 19:47:12 -07:00
|
|
|
|
2021-12-21 16:23:24 -08:00
|
|
|
if ( messageValid ) {
|
2021-12-28 18:49:56 -08:00
|
|
|
state->msgID = stuff.msgID;
|
|
|
|
CNO_FMT( cbuf, stuff.channelNo );
|
|
|
|
XP_LOGFF( TAGFMT() "rcd on %s: msgID=%d, lastMsgRcd=%d ",
|
|
|
|
TAGPRMS, cbuf, stuff.msgID, stuff.lastMsgRcd );
|
|
|
|
} else {
|
2022-09-17 19:47:12 -07:00
|
|
|
XP_LOGFF( TAGFMT() "got message to self?", TAGPRMS ); /* firing */
|
2021-12-28 18:49:56 -08:00
|
|
|
}
|
2021-12-21 16:23:24 -08:00
|
|
|
|
2021-12-28 18:49:56 -08:00
|
|
|
AddressRecord* rec = NULL;
|
|
|
|
XP_U16 streamSize = stream_getSize( stream ); /* anything left? */
|
|
|
|
if ( messageValid ) {
|
|
|
|
if ( stuff.connID == CONN_ID_NONE ) {
|
|
|
|
/* special case: initial message from client or server */
|
2023-02-15 08:31:51 -08:00
|
|
|
rec = validateInitialMessage( comms, streamSize > 0, retAddr,
|
2022-09-06 12:16:29 -07:00
|
|
|
senderID, &stuff.channelNo,
|
2022-10-01 08:43:10 -07:00
|
|
|
stuff.flags, stuff.msgID );
|
2021-12-28 18:49:56 -08:00
|
|
|
state->rec = rec;
|
|
|
|
} else if ( comms->connID == stuff.connID ) {
|
2022-09-06 12:16:29 -07:00
|
|
|
rec = validateChannelMessage( comms, xwe, retAddr,
|
|
|
|
stuff.channelNo, senderID,
|
|
|
|
stuff.msgID, stuff.lastMsgRcd );
|
2014-11-06 06:35:28 -08:00
|
|
|
} else {
|
2021-12-28 18:49:56 -08:00
|
|
|
XP_LOGFF( TAGFMT() "unexpected connID (%x vs %x) ; "
|
2022-09-06 12:16:29 -07:00
|
|
|
"dropping message", TAGPRMS, comms->connID,
|
|
|
|
stuff.connID );
|
2014-11-06 06:35:28 -08:00
|
|
|
}
|
2021-12-28 18:49:56 -08:00
|
|
|
}
|
2006-10-07 03:37:40 +00:00
|
|
|
|
2021-12-28 18:49:56 -08:00
|
|
|
messageValid = messageValid && (NULL != rec)
|
|
|
|
&& (0 == rec->lastMsgRcd || rec->lastMsgRcd <= stuff.msgID);
|
|
|
|
if ( messageValid ) {
|
|
|
|
CNO_FMT( cbuf, stuff.channelNo );
|
|
|
|
XP_LOGFF( TAGFMT() "got %s; msgID=%d; len=%d", TAGPRMS, cbuf,
|
|
|
|
stuff.msgID, streamSize );
|
|
|
|
state->channelNo = stuff.channelNo;
|
|
|
|
comms->lastSaveToken = 0; /* lastMsgRcd no longer valid */
|
|
|
|
stream_setAddress( stream, stuff.channelNo );
|
|
|
|
messageValid = streamSize > 0;
|
|
|
|
resetBackoff( comms );
|
2003-11-01 05:35:29 +00:00
|
|
|
}
|
|
|
|
}
|
2007-12-05 06:33:37 +00:00
|
|
|
|
2014-11-06 06:35:28 -08:00
|
|
|
/* Call after we've had a chance to create rec for addr */
|
|
|
|
noteHBReceived( comms/* , addr */ );
|
2007-12-05 06:33:37 +00:00
|
|
|
|
2014-11-06 06:35:28 -08:00
|
|
|
}
|
2021-03-19 17:25:51 -07:00
|
|
|
#ifdef COMMS_CHECKSUM
|
2021-03-15 16:58:17 -07:00
|
|
|
LOG_RETURNF( "%s (len: %d; sum: %s)", boolToStr(messageValid), state->len, state->sum );
|
2021-03-19 17:25:51 -07:00
|
|
|
#else
|
|
|
|
LOG_RETURNF( "%s (len: %d)", boolToStr(messageValid), state->len );
|
|
|
|
#endif
|
2023-01-06 07:52:41 -08:00
|
|
|
THREAD_CHECK_END();
|
2009-08-21 12:00:09 +00:00
|
|
|
return messageValid;
|
2005-03-09 15:18:17 +00:00
|
|
|
} /* comms_checkIncomingStream */
|
2003-11-01 05:35:29 +00:00
|
|
|
|
2015-02-07 13:24:09 -08:00
|
|
|
void
|
2020-04-25 17:47:07 -07:00
|
|
|
comms_msgProcessed( CommsCtxt* comms, XWEnv xwe,
|
|
|
|
CommsMsgState* state, XP_Bool rejected )
|
2015-02-07 13:24:09 -08:00
|
|
|
{
|
2023-01-06 07:52:41 -08:00
|
|
|
THREAD_CHECK_START(comms);
|
2023-03-16 07:41:53 -07:00
|
|
|
assertQueueOk( comms );
|
2021-03-15 16:58:17 -07:00
|
|
|
#ifdef COMMS_CHECKSUM
|
2022-10-01 08:43:10 -07:00
|
|
|
XP_LOGFF( "rec: %p; len: %d; sum: %s; id: %d; rejected: %s", state->rec,
|
|
|
|
state->len, state->sum, state->msgID, boolToStr(rejected) );
|
2021-03-15 16:58:17 -07:00
|
|
|
#endif
|
2015-02-08 10:37:54 -08:00
|
|
|
XP_ASSERT( comms == state->comms );
|
2015-02-08 07:51:08 -08:00
|
|
|
XP_ASSERT( comms->processingMsg );
|
|
|
|
|
2019-11-03 08:35:56 +00:00
|
|
|
if ( rejected ) {
|
|
|
|
if ( !!state->rec ) {
|
2022-10-01 08:43:10 -07:00
|
|
|
XP_LOGFF( "should I remove rec???; msgID: %d", state->msgID );
|
|
|
|
XP_ASSERT( 1 >= state->msgID );
|
|
|
|
/* this is likely a mistake!!! Why remove it??? */
|
|
|
|
// removeAddrRec( comms, xwe, state->rec );
|
2019-11-03 08:35:56 +00:00
|
|
|
}
|
|
|
|
#ifdef LOG_COMMS_MSGNOS
|
2020-03-14 14:47:26 -07:00
|
|
|
XP_LOGFF( "msg rejected; NOT upping lastMsgRcd to %d", state->msgID );
|
2019-11-03 08:35:56 +00:00
|
|
|
#endif
|
|
|
|
} else {
|
2023-02-15 08:31:51 -08:00
|
|
|
AddressRecord* rec = getRecordFor( comms, state->channelNo );
|
2019-11-03 08:35:56 +00:00
|
|
|
XP_ASSERT( !!rec );
|
|
|
|
if ( !!rec && rec->lastMsgRcd < state->msgID ) {
|
|
|
|
#ifdef LOG_COMMS_MSGNOS
|
2020-03-14 14:47:26 -07:00
|
|
|
XP_LOGFF( "upping lastMsgRcd from %d to %d", rec->lastMsgRcd, state->msgID );
|
2019-11-03 08:35:56 +00:00
|
|
|
#endif
|
|
|
|
rec->lastMsgRcd = state->msgID;
|
|
|
|
}
|
2022-10-07 09:23:09 -07:00
|
|
|
// XP_LOGFF( "CALLING nukeInvites(); might be wrong" );
|
2022-09-08 09:57:11 -07:00
|
|
|
nukeInvites( comms, xwe, state->channelNo );
|
2015-02-07 13:24:09 -08:00
|
|
|
}
|
2015-02-08 07:51:08 -08:00
|
|
|
|
|
|
|
#ifdef DEBUG
|
|
|
|
comms->processingMsg = XP_FALSE;
|
|
|
|
#endif
|
2023-01-06 07:52:41 -08:00
|
|
|
THREAD_CHECK_END();
|
2015-02-07 13:24:09 -08:00
|
|
|
}
|
|
|
|
|
2009-09-05 13:08:46 +00:00
|
|
|
XP_Bool
|
|
|
|
comms_checkComplete( const CommsAddrRec* addr )
|
|
|
|
{
|
|
|
|
XP_Bool result;
|
|
|
|
|
2014-09-25 20:05:37 -07:00
|
|
|
switch ( addr_getType( addr ) ) {
|
2022-09-08 09:57:11 -07:00
|
|
|
#ifdef XWFEATURE_RELAY
|
2009-09-05 13:08:46 +00:00
|
|
|
case COMMS_CONN_RELAY:
|
2009-09-14 03:11:11 +00:00
|
|
|
result = !!addr->u.ip_relay.invite[0]
|
2009-09-05 13:08:46 +00:00
|
|
|
&& !!addr->u.ip_relay.hostName[0]
|
|
|
|
&& !!addr->u.ip_relay.port > 0;
|
|
|
|
break;
|
2022-09-08 09:57:11 -07:00
|
|
|
#endif
|
2009-09-05 13:08:46 +00:00
|
|
|
default:
|
|
|
|
result = XP_TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
2010-10-21 19:14:10 -07:00
|
|
|
XP_Bool
|
|
|
|
comms_canChat( const CommsCtxt* const comms )
|
|
|
|
{
|
2010-11-01 06:57:41 -07:00
|
|
|
XP_Bool canChat = comms_isConnected( comms )
|
2014-12-05 06:19:02 -08:00
|
|
|
&& comms->connID != CONN_ID_NONE
|
2013-07-22 06:12:06 -07:00
|
|
|
&& 64 > comms->queueLen;
|
2010-10-21 19:14:10 -07:00
|
|
|
return canChat;
|
|
|
|
}
|
|
|
|
|
2010-10-27 20:50:11 -07:00
|
|
|
XP_Bool
|
|
|
|
comms_isConnected( const CommsCtxt* const comms )
|
|
|
|
{
|
|
|
|
XP_Bool result = XP_FALSE;
|
2014-10-27 07:51:25 -07:00
|
|
|
CommsConnType typ;
|
2022-09-03 19:14:40 -07:00
|
|
|
for ( XP_U32 st = 0; !result && addr_iter( &comms->selfAddr, &typ, &st ); ) {
|
2020-05-20 13:58:53 -07:00
|
|
|
XP_Bool expected = XP_FALSE;
|
2014-10-27 07:51:25 -07:00
|
|
|
switch ( typ ) {
|
|
|
|
case COMMS_CONN_RELAY:
|
|
|
|
result = 0 != comms->rr.connName[0];
|
2020-05-20 13:58:53 -07:00
|
|
|
expected = XP_TRUE;
|
2014-10-27 07:51:25 -07:00
|
|
|
break;
|
|
|
|
case COMMS_CONN_SMS:
|
|
|
|
case COMMS_CONN_BT:
|
2016-11-14 08:06:53 -08:00
|
|
|
case COMMS_CONN_P2P:
|
2020-05-20 13:58:53 -07:00
|
|
|
case COMMS_CONN_MQTT:
|
|
|
|
expected = XP_TRUE;
|
2014-10-27 07:51:25 -07:00
|
|
|
default:
|
2020-05-20 13:58:53 -07:00
|
|
|
result = comms->connID != CONN_ID_NONE;
|
2014-10-27 07:51:25 -07:00
|
|
|
break;
|
|
|
|
}
|
2020-05-20 13:58:53 -07:00
|
|
|
if ( ! expected ) {
|
|
|
|
XP_LOGFF( "unexpected type %s", ConnType2Str(typ) );
|
|
|
|
}
|
2010-10-27 20:50:11 -07:00
|
|
|
}
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
2023-11-05 11:52:48 -08:00
|
|
|
XP_Bool
|
2023-11-06 11:20:00 -08:00
|
|
|
comms_setQuashed( CommsCtxt* comms, XWEnv xwe, XP_Bool quashed )
|
2023-11-05 11:52:48 -08:00
|
|
|
{
|
|
|
|
XP_U8 flags = comms->flags;
|
|
|
|
if ( quashed ) {
|
|
|
|
flags |= FLAG_QUASHED;
|
|
|
|
} else {
|
|
|
|
flags &= ~FLAG_QUASHED;
|
|
|
|
}
|
|
|
|
XP_Bool changed = flags != comms->flags;
|
|
|
|
if ( changed ) {
|
|
|
|
comms->flags = flags;
|
|
|
|
XP_LOGFF( "(quashed=%s): changing state", boolToStr(quashed) );
|
2023-11-06 11:20:00 -08:00
|
|
|
notifyQueueChanged( comms, xwe );
|
2023-11-05 11:52:48 -08:00
|
|
|
}
|
|
|
|
return changed;
|
|
|
|
}
|
|
|
|
|
2020-09-26 10:12:29 -07:00
|
|
|
#ifdef XWFEATURE_KNOWNPLAYERS
|
2020-09-21 11:31:49 -07:00
|
|
|
void
|
2020-09-25 10:17:47 -07:00
|
|
|
comms_gatherPlayers( CommsCtxt* comms, XWEnv xwe, XP_U32 created )
|
2020-09-21 11:31:49 -07:00
|
|
|
{
|
2023-01-06 07:52:41 -08:00
|
|
|
THREAD_CHECK_START(comms);
|
2020-09-21 11:31:49 -07:00
|
|
|
if ( 0 == (comms->flags & FLAG_HARVEST_DONE) ) {
|
|
|
|
CommsAddrRec addrs[4] = {{0}};
|
|
|
|
XP_U16 nRecs = VSIZE(addrs);
|
2023-02-03 22:00:47 -08:00
|
|
|
comms_getAddrs( comms, addrs, &nRecs );
|
2020-09-21 11:31:49 -07:00
|
|
|
|
|
|
|
const CurGameInfo* gi = comms->util->gameInfo;
|
2020-10-23 16:37:28 -07:00
|
|
|
XP_ASSERT( 0 < gi->nPlayers );
|
2020-09-25 10:17:47 -07:00
|
|
|
if ( kplr_addAddrs( comms->dutil, xwe, gi, addrs, nRecs, created ) ) {
|
2023-03-06 21:10:09 -08:00
|
|
|
/* if ( 1 ) { */
|
|
|
|
/* XP_LOGFF( "not setting flag :-)" ); */
|
|
|
|
/* } else { */
|
|
|
|
/* /\* Need a way to force/override this manually? *\/ */
|
|
|
|
/* comms->flags |= FLAG_HARVEST_DONE; */
|
|
|
|
/* } */
|
2020-09-21 11:31:49 -07:00
|
|
|
}
|
|
|
|
}
|
2023-01-06 07:52:41 -08:00
|
|
|
THREAD_CHECK_END();
|
2023-03-06 21:10:09 -08:00
|
|
|
LOG_RETURN_VOID();
|
2020-09-21 11:31:49 -07:00
|
|
|
}
|
2020-09-26 10:12:29 -07:00
|
|
|
#endif
|
2020-09-21 11:31:49 -07:00
|
|
|
|
2017-11-10 21:34:02 -08:00
|
|
|
#ifdef RELAY_VIA_HTTP
|
|
|
|
void
|
2020-04-25 17:47:07 -07:00
|
|
|
comms_gameJoined( CommsCtxt* comms, XWEnv xwe, const XP_UCHAR* connname, XWHostID hid )
|
2017-11-10 21:34:02 -08:00
|
|
|
{
|
|
|
|
LOG_FUNC();
|
|
|
|
XP_ASSERT( XP_STRLEN( connname ) + 1 < sizeof(comms->rr.connName) );
|
|
|
|
XP_STRNCPY( comms->rr.connName, connname, sizeof(comms->rr.connName) );
|
|
|
|
comms->rr.myHostID = hid;
|
|
|
|
comms->forceChannel = hid;
|
2020-04-25 17:47:07 -07:00
|
|
|
set_relay_state( comms, xwe, COMMS_RELAYSTATE_USING_HTTP );
|
2017-11-10 21:34:02 -08:00
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2012-04-13 20:21:59 -07:00
|
|
|
#if defined COMMS_HEARTBEAT || defined XWFEATURE_COMMSACK
|
2007-12-05 06:33:37 +00:00
|
|
|
static void
|
2023-10-26 12:25:59 -07:00
|
|
|
sendEmptyMsg( CommsCtxt* comms, XWEnv xwe, AddressRecord* rec )
|
2007-12-05 06:33:37 +00:00
|
|
|
{
|
2023-10-26 12:25:59 -07:00
|
|
|
THREAD_CHECK_START(comms);
|
2022-09-06 12:16:29 -07:00
|
|
|
MsgQueueElem* elem = makeElemWithID( comms, xwe, 0 /* msgID */,
|
|
|
|
rec, rec? rec->channelNo : 0, NULL );
|
2023-10-26 12:25:59 -07:00
|
|
|
XP_ASSERT( !!elem );
|
|
|
|
elem = addToQueue( comms, xwe, elem, XP_FALSE );
|
|
|
|
if ( !!elem ) {
|
|
|
|
sendMsg( comms, xwe, elem, COMMS_CONN_NONE );
|
|
|
|
}
|
|
|
|
THREAD_CHECK_END();
|
2007-12-05 06:33:37 +00:00
|
|
|
} /* sendEmptyMsg */
|
2012-04-13 20:21:59 -07:00
|
|
|
#endif
|
2007-12-05 06:33:37 +00:00
|
|
|
|
2012-04-13 20:21:59 -07:00
|
|
|
#ifdef COMMS_HEARTBEAT
|
2007-11-18 23:43:27 +00: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 06:33:37 +00:00
|
|
|
do {
|
|
|
|
if ( comms->lastMsgRcvdTime > 0 ) {
|
2020-04-26 11:28:18 -07:00
|
|
|
XP_U32 now = dutil_getCurSeconds( comms->dutil, xwe );
|
2007-12-05 06:33:37 +00:00
|
|
|
XP_U32 tooLongAgo = now - (HB_INTERVAL * 2);
|
|
|
|
if ( comms->lastMsgRcvdTime < tooLongAgo ) {
|
2022-04-29 09:19:44 -07:00
|
|
|
XP_LOGFF( "calling reset proc; last was %ld secs too long "
|
|
|
|
"ago", tooLongAgo - comms->lastMsgRcvdTime );
|
2009-09-12 21:39:13 +00:00
|
|
|
(*comms->procs.reset)(comms->procs.closure);
|
2007-12-05 06:33:37 +00:00
|
|
|
comms->lastMsgRcvdTime = 0;
|
|
|
|
break; /* outta here */
|
|
|
|
}
|
2007-11-26 02:58:25 +00:00
|
|
|
}
|
2007-11-18 23:43:27 +00:00
|
|
|
|
2007-12-05 06:33:37 +00:00
|
|
|
if ( comms->recs ) {
|
|
|
|
AddressRecord* rec;
|
|
|
|
for ( rec = comms->recs; !!rec; rec = rec->next ) {
|
|
|
|
sendEmptyMsg( comms, rec );
|
2007-11-26 02:58:25 +00:00
|
|
|
}
|
2007-12-05 06:33:37 +00:00
|
|
|
} else if ( !comms->isServer ) {
|
|
|
|
/* Client still waiting for inital ALL_REG message */
|
|
|
|
sendEmptyMsg( comms, NULL );
|
2007-11-18 23:43:27 +00:00
|
|
|
}
|
2007-12-05 06:33:37 +00:00
|
|
|
} while ( XP_FALSE );
|
2007-11-18 23:43:27 +00:00
|
|
|
|
2007-12-05 06:33:37 +00:00
|
|
|
setHeartbeatTimer( comms );
|
2007-11-18 23:43:27 +00:00
|
|
|
} /* heartbeat_checks */
|
|
|
|
#endif
|
|
|
|
|
2012-04-13 20:21:59 -07:00
|
|
|
#if defined RELAY_HEARTBEAT || defined COMMS_HEARTBEAT
|
2008-04-12 15:36:31 +00:00
|
|
|
static XP_Bool
|
2020-04-25 11:34:58 -07:00
|
|
|
p_comms_timerFired( void* closure, XWEnv xwe, XWTimerReason XP_UNUSED_DBG(why) )
|
2005-06-23 04:16:53 +00:00
|
|
|
{
|
|
|
|
CommsCtxt* comms = (CommsCtxt*)closure;
|
2009-02-01 15:50:58 +00:00
|
|
|
XP_ASSERT( why == TIMER_COMMS );
|
2007-12-05 06:33:37 +00:00
|
|
|
LOG_FUNC();
|
|
|
|
comms->hbTimerPending = XP_FALSE;
|
2007-11-18 23:43:27 +00:00
|
|
|
if (0 ) {
|
2010-09-11 01:23:13 -07:00
|
|
|
#if defined XWFEATURE_RELAY && defined RELAY_HEARTBEAT
|
2022-09-03 19:14:40 -07:00
|
|
|
} else if ( (comms->selfAddr.conType == COMMS_CONN_RELAY )
|
2014-02-07 21:12:45 -08:00
|
|
|
&& (comms->rr.heartbeat != HEARTBEAT_NONE) ) {
|
2020-04-25 17:47:07 -07:00
|
|
|
(void)send_via_relay( comms, xwe, XWRELAY_HEARTBEAT, HOST_ID_NONE,
|
2015-09-30 06:50:04 -07:00
|
|
|
NULL, 0, NULL );
|
2005-07-05 20:57:37 +00:00
|
|
|
/* No need to reset timer. send_via_relay does that. */
|
2009-02-01 15:50:58 +00:00
|
|
|
#endif
|
|
|
|
#ifdef COMMS_HEARTBEAT
|
2007-11-18 23:43:27 +00:00
|
|
|
} else {
|
|
|
|
XP_ASSERT( comms->doHeartbeat );
|
|
|
|
heartbeat_checks( comms );
|
|
|
|
#endif
|
2005-06-23 04:16:53 +00:00
|
|
|
}
|
2008-04-12 15:36:31 +00:00
|
|
|
return XP_FALSE; /* no need for redraw */
|
2007-11-18 23:43:27 +00:00
|
|
|
} /* p_comms_timerFired */
|
2005-06-23 04:16:53 +00:00
|
|
|
|
|
|
|
static void
|
|
|
|
setHeartbeatTimer( CommsCtxt* comms )
|
|
|
|
{
|
2009-01-05 02:07:30 +00:00
|
|
|
XP_ASSERT( !!comms );
|
2009-02-01 15:50:58 +00:00
|
|
|
|
|
|
|
if ( comms->hbTimerPending ) {
|
2022-04-29 09:19:44 -07:00
|
|
|
XP_LOGFF( "skipping b/c hbTimerPending" );
|
2009-02-01 15:50:58 +00:00
|
|
|
} else if ( comms->reconTimerPending ) {
|
2022-04-29 09:19:44 -07:00
|
|
|
XP_LOGFF( "skipping b/c reconTimerPending" );
|
2009-02-01 15:50:58 +00:00
|
|
|
} else {
|
2007-12-05 06:33:37 +00:00
|
|
|
XP_U16 when = 0;
|
2009-02-01 15:50:58 +00:00
|
|
|
#ifdef XWFEATURE_RELAY
|
2022-09-03 19:14:40 -07:00
|
|
|
if ( comms->selfAddr.conType == COMMS_CONN_RELAY ) {
|
2014-02-07 21:12:45 -08:00
|
|
|
when = comms->rr.heartbeat;
|
2007-12-05 06:33:37 +00:00
|
|
|
}
|
2009-02-01 15:50:58 +00:00
|
|
|
#endif
|
|
|
|
#ifdef COMMS_HEARTBEAT
|
2007-12-05 06:33:37 +00:00
|
|
|
if ( comms->doHeartbeat ) {
|
2022-04-29 09:19:44 -07:00
|
|
|
XP_LOGFF( "calling util_setTimer" );
|
2007-12-05 06:33:37 +00:00
|
|
|
when = HB_INTERVAL;
|
|
|
|
}
|
2007-11-18 23:43:27 +00:00
|
|
|
#endif
|
2007-12-05 06:33:37 +00:00
|
|
|
if ( when != 0 ) {
|
2020-04-26 11:28:18 -07:00
|
|
|
util_setTimer( comms->util, xwe, TIMER_COMMS, when,
|
2007-12-05 06:33:37 +00:00
|
|
|
p_comms_timerFired, comms );
|
|
|
|
comms->hbTimerPending = XP_TRUE;
|
|
|
|
}
|
|
|
|
}
|
2007-11-18 23:43:27 +00:00
|
|
|
} /* setHeartbeatTimer */
|
2005-07-23 15:28:15 +00:00
|
|
|
#endif
|
2005-06-23 04:16:53 +00:00
|
|
|
|
2009-11-22 18:00:20 +00:00
|
|
|
const char*
|
|
|
|
ConnType2Str( CommsConnType typ )
|
|
|
|
{
|
|
|
|
switch( typ ) {
|
|
|
|
CASESTR(COMMS_CONN_NONE);
|
|
|
|
CASESTR( COMMS_CONN_IR );
|
|
|
|
CASESTR( COMMS_CONN_IP_DIRECT );
|
|
|
|
CASESTR( COMMS_CONN_RELAY );
|
|
|
|
CASESTR( COMMS_CONN_BT );
|
|
|
|
CASESTR( COMMS_CONN_SMS );
|
2016-11-14 08:06:53 -08:00
|
|
|
CASESTR( COMMS_CONN_P2P );
|
2014-09-25 20:05:37 -07:00
|
|
|
CASESTR( COMMS_CONN_NTYPES );
|
2019-11-23 10:53:51 -08:00
|
|
|
CASESTR( COMMS_CONN_NFC );
|
2020-05-20 13:58:53 -07:00
|
|
|
CASESTR( COMMS_CONN_MQTT );
|
2009-11-22 18:00:20 +00:00
|
|
|
default:
|
|
|
|
XP_ASSERT(0);
|
|
|
|
}
|
|
|
|
return "<unknown>";
|
|
|
|
} /* ConnType2Str */
|
|
|
|
|
2021-03-19 13:03:39 -07:00
|
|
|
#ifdef DEBUG
|
2023-02-01 19:08:04 -08:00
|
|
|
|
|
|
|
static ForEachAct
|
|
|
|
statsProc( MsgQueueElem* elem, void* closure )
|
|
|
|
{
|
|
|
|
XWStreamCtxt* stream = (XWStreamCtxt*)closure;
|
|
|
|
XP_UCHAR buf[100];
|
|
|
|
XP_SNPRINTF( buf, sizeof(buf),
|
|
|
|
"msgID: " XP_LD ": channelNo=%.4X; len=%d\n",
|
2023-03-18 15:15:34 -07:00
|
|
|
elem->msgID, elem->channelNo, elem->smp.len );
|
2023-02-01 19:08:04 -08:00
|
|
|
stream_catString( stream, buf );
|
|
|
|
return FEA_OK;
|
|
|
|
}
|
|
|
|
|
2003-11-01 05:35:29 +00:00
|
|
|
void
|
2022-12-26 19:24:39 -08:00
|
|
|
comms_getStats( const CommsCtxt* comms, XWStreamCtxt* stream )
|
2003-11-01 05:35:29 +00:00
|
|
|
{
|
|
|
|
XP_UCHAR buf[100];
|
2020-08-23 20:09:21 -07:00
|
|
|
|
2006-09-27 01:54:53 +00:00
|
|
|
XP_SNPRINTF( (XP_UCHAR*)buf, sizeof(buf),
|
2023-11-05 11:52:48 -08:00
|
|
|
(XP_UCHAR*)"role: %s; msg queue len: %d; quashed: %s;\n",
|
2020-08-25 14:47:56 -07:00
|
|
|
comms->isServer ? "host" : "guest",
|
2023-11-05 11:52:48 -08:00
|
|
|
comms->queueLen, boolToStr(QUASHED(comms)) );
|
2009-03-29 18:26:59 +00:00
|
|
|
stream_catString( stream, buf );
|
2006-09-27 01:54:53 +00:00
|
|
|
|
2023-02-01 19:08:04 -08:00
|
|
|
forEachElem( (CommsCtxt*)comms, statsProc, stream );
|
2003-11-01 05:35:29 +00:00
|
|
|
|
2020-08-23 20:09:21 -07:00
|
|
|
for ( AddressRecord* rec = comms->recs; !!rec; rec = rec->next ) {
|
2019-12-23 08:45:55 -08:00
|
|
|
XP_SNPRINTF( (XP_UCHAR*)buf, sizeof(buf),
|
2022-12-27 16:03:08 -08:00
|
|
|
(XP_UCHAR*)"Stats for channel %.4X msgs\n",
|
2003-11-01 05:35:29 +00:00
|
|
|
rec->channelNo );
|
2009-03-29 18:26:59 +00:00
|
|
|
stream_catString( stream, buf );
|
2003-11-01 05:35:29 +00:00
|
|
|
|
2019-12-23 08:45:55 -08:00
|
|
|
XP_SNPRINTF( (XP_UCHAR*)buf, sizeof(buf),
|
2022-12-27 16:03:08 -08:00
|
|
|
(XP_UCHAR*)" Last sent: " XP_LD "; ",
|
2003-11-01 05:35:29 +00:00
|
|
|
rec->nextMsgID );
|
2009-03-29 18:26:59 +00:00
|
|
|
stream_catString( stream, buf );
|
2003-11-01 05:35:29 +00:00
|
|
|
|
2019-12-23 08:45:55 -08:00
|
|
|
XP_SNPRINTF( (XP_UCHAR*)buf, sizeof(buf),
|
2022-12-27 16:03:08 -08:00
|
|
|
(XP_UCHAR*)"last rcvd: %d\n",
|
2010-11-03 18:28:54 -07:00
|
|
|
rec->lastMsgRcd );
|
|
|
|
stream_catString( stream, buf );
|
2003-11-01 05:35:29 +00:00
|
|
|
}
|
|
|
|
} /* comms_getStats */
|
2014-11-06 06:35:28 -08:00
|
|
|
|
|
|
|
void
|
|
|
|
comms_setAddrDisabled( CommsCtxt* comms, CommsConnType typ,
|
|
|
|
XP_Bool send, XP_Bool disabled )
|
|
|
|
{
|
|
|
|
XP_ASSERT( !!comms );
|
2022-04-29 09:19:44 -07:00
|
|
|
XP_LOGFF( "(typ=%s, send=%d, disabled=%d)",
|
2014-11-06 06:35:28 -08:00
|
|
|
ConnType2Str(typ), send, disabled );
|
|
|
|
comms->disableds[typ][send?0:1] = disabled;
|
|
|
|
}
|
|
|
|
|
|
|
|
XP_Bool
|
|
|
|
comms_getAddrDisabled( const CommsCtxt* comms, CommsConnType typ,
|
|
|
|
XP_Bool send )
|
|
|
|
{
|
|
|
|
XP_ASSERT( !!comms );
|
|
|
|
return comms->disableds[typ][send?0:1];
|
|
|
|
}
|
2003-11-01 05:35:29 +00:00
|
|
|
#endif
|
|
|
|
|
|
|
|
static AddressRecord*
|
2023-02-15 08:31:51 -08:00
|
|
|
rememberChannelAddress( CommsCtxt* comms, XP_PlayerAddr channelNo,
|
2021-12-21 16:23:24 -08:00
|
|
|
XWHostID hostID, const CommsAddrRec* addr, XP_U16 flags )
|
2003-11-01 05:35:29 +00:00
|
|
|
{
|
2023-01-06 07:52:41 -08:00
|
|
|
AddressRecord* rec = NULL;
|
|
|
|
THREAD_CHECK_START( comms );
|
2014-11-06 18:44:19 -08:00
|
|
|
CNO_FMT( cbuf, channelNo );
|
2022-04-29 09:19:44 -07:00
|
|
|
XP_LOGFF( "(%s)", cbuf );
|
2022-10-07 09:23:09 -07:00
|
|
|
listRecs( comms, "entering rememberChannelAddress" );
|
2014-11-06 20:28:51 -08:00
|
|
|
|
2023-02-16 21:36:46 -08:00
|
|
|
logAddrComms( comms, addr, __func__ );
|
2023-02-15 08:31:51 -08:00
|
|
|
rec = getRecordFor( comms, channelNo );
|
2014-11-05 07:41:20 -08:00
|
|
|
if ( !rec ) {
|
2005-03-06 17:56:34 +00:00
|
|
|
/* not found; add a new entry */
|
2014-11-05 07:41:20 -08:00
|
|
|
rec = (AddressRecord*)XP_CALLOC( comms->mpool, sizeof(*rec) );
|
2005-03-06 17:56:34 +00:00
|
|
|
|
2014-11-05 07:41:20 -08:00
|
|
|
rec->channelNo = channelNo;
|
|
|
|
rec->rr.hostID = hostID;
|
2021-12-29 07:35:43 -08:00
|
|
|
rec->flags = flags & VERSION_MASK;
|
2021-12-21 16:23:24 -08:00
|
|
|
|
2014-11-05 07:41:20 -08:00
|
|
|
rec->next = comms->recs;
|
|
|
|
comms->recs = rec;
|
2021-12-21 16:23:24 -08:00
|
|
|
XP_LOGFF( "creating rec %p for %s, hostID = %d, flags=0x%x",
|
|
|
|
rec, cbuf, hostID, flags );
|
2005-03-06 17:56:34 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/* overwrite existing address with new one. I assume that's the right
|
|
|
|
move. */
|
2014-11-05 07:41:20 -08:00
|
|
|
if ( !!rec ) {
|
2005-03-06 17:56:34 +00:00
|
|
|
if ( !!addr ) {
|
2022-04-29 09:19:44 -07:00
|
|
|
XP_LOGFF( "replacing/adding addr with _conTypes %x with %x",
|
2014-11-05 07:41:20 -08:00
|
|
|
rec->addr._conTypes, addr->_conTypes );
|
|
|
|
XP_MEMCPY( &rec->addr, addr, sizeof(rec->addr) );
|
|
|
|
XP_ASSERT( rec->rr.hostID == hostID );
|
2005-03-06 17:56:34 +00:00
|
|
|
} else {
|
2022-04-29 09:19:44 -07:00
|
|
|
XP_LOGFF( "storing addr with _conTypes %x",
|
2014-11-05 07:41:20 -08:00
|
|
|
addr->_conTypes );
|
|
|
|
XP_MEMSET( &rec->addr, 0, sizeof(rec->addr) );
|
2022-09-03 19:14:40 -07:00
|
|
|
rec->addr._conTypes = comms->selfAddr._conTypes;
|
|
|
|
// addr_setTypes( &recs->addr, addr_getTypes( &comms->selfAddr ) );
|
2003-11-01 05:35:29 +00:00
|
|
|
}
|
|
|
|
}
|
2023-11-17 19:39:45 -08:00
|
|
|
listRecs( comms, "leaving rememberChannelAddress()" );
|
2023-01-06 07:52:41 -08:00
|
|
|
THREAD_CHECK_END();
|
2014-11-05 07:41:20 -08:00
|
|
|
return rec;
|
2003-11-01 05:35:29 +00:00
|
|
|
} /* rememberChannelAddress */
|
|
|
|
|
2014-11-05 07:41:20 -08:00
|
|
|
#ifdef DEBUG
|
2023-02-16 21:36:46 -08:00
|
|
|
|
|
|
|
static void
|
|
|
|
logAddrComms( const CommsCtxt* comms, const CommsAddrRec* addr,
|
|
|
|
const char* caller )
|
|
|
|
{
|
2023-12-18 17:08:51 -08:00
|
|
|
logAddr( comms->dutil, addr, caller );
|
2023-02-16 21:36:46 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
void
|
2023-12-18 17:08:51 -08:00
|
|
|
logAddr( XW_DUtilCtxt* dutil, const CommsAddrRec* addr,
|
2023-02-03 22:00:47 -08:00
|
|
|
const char* caller )
|
2014-11-05 07:41:20 -08:00
|
|
|
{
|
|
|
|
if ( !!addr ) {
|
|
|
|
char buf[128];
|
2023-12-18 17:08:51 -08:00
|
|
|
XWStreamCtxt* stream = mem_stream_make_raw( MPPARM(dutil->mpool)
|
2023-02-16 21:36:46 -08:00
|
|
|
dutil_getVTManager(dutil));
|
|
|
|
if ( !!caller ) {
|
|
|
|
snprintf( buf, sizeof(buf), "called on %p from %s:\n",
|
|
|
|
addr, caller );
|
|
|
|
stream_catString( stream, buf );
|
|
|
|
}
|
2014-11-05 07:41:20 -08:00
|
|
|
|
|
|
|
CommsConnType typ;
|
|
|
|
XP_Bool first = XP_TRUE;
|
|
|
|
for ( XP_U32 st = 0; addr_iter( addr, &typ, &st ); ) {
|
|
|
|
if ( !first ) {
|
|
|
|
stream_catString( stream, "\n" );
|
|
|
|
}
|
|
|
|
|
|
|
|
snprintf( buf, sizeof(buf), "* %s: ", ConnType2Str(typ) );
|
|
|
|
stream_catString( stream, buf );
|
|
|
|
|
|
|
|
switch( typ ) {
|
|
|
|
case COMMS_CONN_RELAY:
|
2022-10-25 21:02:18 -07:00
|
|
|
#ifdef XWFEATURE_RELAY
|
2014-11-05 07:41:20 -08:00
|
|
|
stream_catString( stream, "room: " );
|
|
|
|
stream_catString( stream, addr->u.ip_relay.invite );
|
|
|
|
stream_catString( stream, "; host: " );
|
|
|
|
stream_catString( stream, addr->u.ip_relay.hostName );
|
2022-09-08 09:57:11 -07:00
|
|
|
#endif
|
2022-10-25 21:02:18 -07:00
|
|
|
break;
|
2014-11-05 07:41:20 -08:00
|
|
|
case COMMS_CONN_SMS:
|
|
|
|
stream_catString( stream, "phone: " );
|
|
|
|
stream_catString( stream, addr->u.sms.phone );
|
|
|
|
stream_catString( stream, "; port: " );
|
|
|
|
snprintf( buf, sizeof(buf), "%d", addr->u.sms.port );
|
|
|
|
stream_catString( stream, buf );
|
|
|
|
break;
|
2014-11-13 07:27:03 -08:00
|
|
|
case COMMS_CONN_BT:
|
|
|
|
stream_catString( stream, "host: " );
|
|
|
|
stream_catString( stream, addr->u.bt.hostName );
|
2014-11-24 06:30:58 -08:00
|
|
|
stream_catString( stream, "; addr: " );
|
2014-11-13 07:27:03 -08:00
|
|
|
stream_catString( stream, addr->u.bt.btAddr.chars );
|
|
|
|
break;
|
2016-11-14 08:06:53 -08:00
|
|
|
case COMMS_CONN_P2P:
|
|
|
|
stream_catString( stream, "mac addr: " );
|
|
|
|
stream_catString( stream, addr->u.p2p.mac_addr );
|
|
|
|
break;
|
2019-11-23 10:53:51 -08:00
|
|
|
case COMMS_CONN_NFC:
|
|
|
|
break;
|
2020-05-20 13:58:53 -07:00
|
|
|
case COMMS_CONN_MQTT: {
|
|
|
|
stream_catString( stream, "mqtt devID: " );
|
|
|
|
XP_UCHAR buf[32];
|
|
|
|
XP_SNPRINTF( buf, VSIZE(buf), MQTTDevID_FMT, addr->u.mqtt.devID );
|
|
|
|
stream_catString( stream, buf );
|
|
|
|
}
|
|
|
|
break;
|
2014-11-05 07:41:20 -08:00
|
|
|
default:
|
|
|
|
XP_ASSERT(0);
|
|
|
|
}
|
|
|
|
first = XP_FALSE;
|
|
|
|
}
|
|
|
|
stream_putU8( stream, '\0' );
|
2022-04-29 09:19:44 -07:00
|
|
|
XP_LOGFF( "%s", stream_getPtr( stream ) );
|
2023-02-03 22:00:47 -08:00
|
|
|
stream_destroy( stream );
|
2014-11-05 07:41:20 -08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
static void
|
2020-09-10 15:11:22 -07:00
|
|
|
augmentChannelAddr( CommsCtxt* comms, AddressRecord* const rec,
|
|
|
|
const CommsAddrRec* addr, XWHostID hostID )
|
2020-02-24 21:10:49 +01:00
|
|
|
{
|
2023-01-06 07:52:41 -08:00
|
|
|
THREAD_CHECK_START( comms );
|
2020-09-25 10:17:47 -07:00
|
|
|
augmentAddrIntrnl( comms, &rec->addr, addr, XP_TRUE );
|
2022-10-19 20:33:30 -07:00
|
|
|
#ifdef XWFEATURE_RELAY
|
2020-02-24 21:10:49 +01:00
|
|
|
if ( addr_hasType( &rec->addr, COMMS_CONN_RELAY ) ) {
|
|
|
|
if ( 0 != hostID ) {
|
|
|
|
rec->rr.hostID = hostID;
|
2022-04-29 09:19:44 -07:00
|
|
|
XP_LOGFF( "set hostID for rec %p to %d", rec, hostID );
|
2020-02-24 21:10:49 +01:00
|
|
|
}
|
|
|
|
}
|
2022-10-19 20:33:30 -07:00
|
|
|
#else
|
|
|
|
XP_USE(hostID);
|
|
|
|
#endif
|
2020-09-10 15:11:22 -07:00
|
|
|
|
|
|
|
#ifdef DEBUG
|
|
|
|
CommsConnType typ;
|
|
|
|
for ( XP_U32 st = 0; addr_iter( addr, &typ, &st ); ) {
|
2022-09-03 19:14:40 -07:00
|
|
|
if ( !addr_hasType( &comms->selfAddr, typ ) ) {
|
2020-09-10 15:11:22 -07:00
|
|
|
XP_LOGFF( "main addr missing type %s", ConnType2Str(typ) );
|
|
|
|
}
|
|
|
|
}
|
|
|
|
#endif
|
2023-01-06 07:52:41 -08:00
|
|
|
THREAD_CHECK_END();
|
2020-02-24 21:10:49 +01:00
|
|
|
}
|
|
|
|
|
2020-09-24 16:37:50 -07:00
|
|
|
static XP_Bool
|
|
|
|
augmentAddrIntrnl( CommsCtxt* comms, CommsAddrRec* destAddr,
|
2020-09-25 10:17:47 -07:00
|
|
|
const CommsAddrRec* srcAddr, XP_Bool isNewer )
|
2014-11-05 07:41:20 -08:00
|
|
|
{
|
2020-09-24 16:37:50 -07:00
|
|
|
XP_Bool changed = XP_FALSE;
|
2023-01-06 07:52:41 -08:00
|
|
|
ASSERT_ADDR_OK( srcAddr );
|
2020-09-25 10:17:47 -07:00
|
|
|
const CommsAddrRec empty = {0};
|
2020-02-24 21:10:49 +01:00
|
|
|
if ( !!srcAddr ) {
|
2014-11-05 07:41:20 -08:00
|
|
|
CommsConnType typ;
|
2020-02-24 21:10:49 +01:00
|
|
|
for ( XP_U32 st = 0; addr_iter( srcAddr, &typ, &st ); ) {
|
2020-09-25 10:17:47 -07:00
|
|
|
XP_Bool newType = !addr_hasType( destAddr, typ );
|
|
|
|
if ( newType ) {
|
2020-02-24 21:10:49 +01:00
|
|
|
XP_LOGFF( "adding new type %s to rec", ConnType2Str(typ) );
|
|
|
|
addr_addType( destAddr, typ );
|
2020-09-10 15:11:22 -07:00
|
|
|
|
|
|
|
/* If an address is getting added to a channel, the top-level
|
|
|
|
address should also include the type. The specifics of the
|
2022-09-03 18:47:04 -07:00
|
|
|
address don't make sense to copy, however.
|
|
|
|
NO -- not any more. I have the addresses my user gives me
|
|
|
|
*/
|
2022-09-03 19:14:40 -07:00
|
|
|
if ( !!comms && ! addr_hasType( &comms->selfAddr, typ ) ) {
|
|
|
|
/* we just added it, so can't be comms->selfAddr */
|
|
|
|
XP_ASSERT( destAddr != &comms->selfAddr );
|
|
|
|
XP_LOGFF( "NOT adding %s to comms->selfAddr", ConnType2Str(typ) );
|
|
|
|
// addr_addType( &comms->selfAddr, typ );
|
2020-09-10 15:11:22 -07:00
|
|
|
}
|
2014-12-05 06:19:02 -08:00
|
|
|
}
|
2014-11-05 07:41:20 -08:00
|
|
|
|
|
|
|
const void* src = NULL;
|
|
|
|
void* dest = NULL;
|
2020-02-24 21:10:49 +01:00
|
|
|
size_t siz;
|
2014-11-05 07:41:20 -08:00
|
|
|
|
|
|
|
switch( typ ) {
|
2022-09-08 09:57:11 -07:00
|
|
|
#ifdef XWFEATURE_RELAY
|
2014-11-05 07:41:20 -08:00
|
|
|
case COMMS_CONN_RELAY:
|
2020-02-24 21:10:49 +01:00
|
|
|
dest = &destAddr->u.ip_relay;
|
|
|
|
src = &srcAddr->u.ip_relay;
|
|
|
|
siz = sizeof( destAddr->u.ip_relay );
|
2014-11-05 07:41:20 -08:00
|
|
|
break;
|
2022-09-08 09:57:11 -07:00
|
|
|
#endif
|
2014-11-05 07:41:20 -08:00
|
|
|
case COMMS_CONN_SMS:
|
2020-02-24 21:10:49 +01:00
|
|
|
XP_ASSERT( 0 != srcAddr->u.sms.port );
|
|
|
|
XP_ASSERT( '\0' != srcAddr->u.sms.phone[0] );
|
|
|
|
dest = &destAddr->u.sms;
|
|
|
|
src = &srcAddr->u.sms;
|
|
|
|
siz = sizeof(destAddr->u.sms);
|
2014-11-05 07:41:20 -08:00
|
|
|
break;
|
2016-11-14 08:06:53 -08:00
|
|
|
case COMMS_CONN_P2P:
|
2020-02-24 21:10:49 +01:00
|
|
|
XP_ASSERT( '\0' != srcAddr->u.p2p.mac_addr[0] );
|
|
|
|
dest = &destAddr->u.p2p;
|
|
|
|
src = &srcAddr->u.p2p;
|
|
|
|
siz = sizeof(destAddr->u.p2p);
|
2016-11-14 08:06:53 -08:00
|
|
|
break;
|
2014-11-14 07:06:02 -08:00
|
|
|
case COMMS_CONN_BT:
|
2021-02-07 16:31:32 -08:00
|
|
|
#ifdef XWFEATURE_BLUETOOTH
|
2020-02-24 21:10:49 +01:00
|
|
|
dest = &destAddr->u.bt;
|
|
|
|
src = &srcAddr->u.bt;
|
|
|
|
siz = sizeof(destAddr->u.bt);
|
2014-11-14 07:06:02 -08:00
|
|
|
#endif
|
2021-02-07 16:31:32 -08:00
|
|
|
break;
|
2019-11-23 10:53:51 -08:00
|
|
|
case COMMS_CONN_NFC:
|
|
|
|
break;
|
2020-05-20 13:58:53 -07:00
|
|
|
case COMMS_CONN_MQTT:
|
2022-09-03 18:47:04 -07:00
|
|
|
XP_ASSERT( 0 != srcAddr->u.mqtt.devID );
|
2020-05-20 13:58:53 -07:00
|
|
|
dest = &destAddr->u.mqtt;
|
|
|
|
src = &srcAddr->u.mqtt;
|
|
|
|
siz = sizeof(destAddr->u.mqtt);
|
|
|
|
break;
|
2014-11-05 07:41:20 -08:00
|
|
|
default:
|
|
|
|
XP_ASSERT(0);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
if ( !!dest ) {
|
2020-09-26 10:12:29 -07:00
|
|
|
XP_ASSERT( !newType || 0 == XP_MEMCMP( &empty, dest, siz ) );
|
|
|
|
|
2020-09-25 10:17:47 -07:00
|
|
|
XP_Bool different = 0 != XP_MEMCMP( dest, src, siz );
|
|
|
|
if ( different ) {
|
|
|
|
/* If the dest is non-empty AND the src is older, don't do
|
|
|
|
anything: don't replace newer info with older. Note
|
|
|
|
that this assumes unset values are empty!!! */
|
|
|
|
if ( !isNewer && !newType
|
|
|
|
&& 0 != XP_MEMCMP( &empty, dest, siz ) ) {
|
|
|
|
XP_LOGFF( "%s: not replacing new info with old",
|
|
|
|
ConnType2Str(typ) );
|
|
|
|
} else {
|
|
|
|
XP_MEMCPY( dest, src, siz );
|
|
|
|
changed = XP_TRUE;
|
2020-09-05 14:56:03 -07:00
|
|
|
}
|
2014-11-11 07:46:01 -08:00
|
|
|
}
|
2014-11-05 07:41:20 -08:00
|
|
|
}
|
2014-12-05 06:19:02 -08:00
|
|
|
}
|
2014-11-05 07:41:20 -08:00
|
|
|
}
|
2020-09-24 16:37:50 -07:00
|
|
|
return changed;
|
|
|
|
}
|
|
|
|
|
|
|
|
XP_Bool
|
2020-09-25 10:17:47 -07:00
|
|
|
augmentAddr( CommsAddrRec* addr, const CommsAddrRec* newer, XP_Bool isNewer )
|
2020-09-24 16:37:50 -07:00
|
|
|
{
|
2020-09-25 10:17:47 -07:00
|
|
|
return augmentAddrIntrnl( NULL, addr, newer, isNewer );
|
2014-11-05 07:41:20 -08:00
|
|
|
}
|
|
|
|
|
2003-11-01 05:35:29 +00:00
|
|
|
static XP_Bool
|
2023-02-15 08:31:51 -08:00
|
|
|
channelToAddress( const CommsCtxt* comms, XP_PlayerAddr channelNo,
|
2007-02-08 02:53:10 +00:00
|
|
|
const CommsAddrRec** addr )
|
2003-11-01 05:35:29 +00:00
|
|
|
{
|
2023-02-15 08:31:51 -08:00
|
|
|
AddressRecord* recs = getRecordFor( comms, channelNo );
|
2007-02-08 02:53:10 +00:00
|
|
|
XP_Bool found = !!recs;
|
|
|
|
*addr = found? &recs->addr : NULL;
|
|
|
|
return found;
|
2003-11-01 05:35:29 +00:00
|
|
|
} /* channelToAddress */
|
|
|
|
|
|
|
|
static XP_U16
|
2006-10-15 13:53:17 +00:00
|
|
|
countAddrRecs( const CommsCtxt* comms )
|
2003-11-01 05:35:29 +00:00
|
|
|
{
|
2022-10-07 09:23:09 -07:00
|
|
|
XP_U16 count = 0;
|
|
|
|
for ( AddressRecord* recs = comms->recs; !!recs; recs = recs->next ) {
|
2005-03-19 22:01:38 +00:00
|
|
|
++count;
|
2003-11-01 05:35:29 +00:00
|
|
|
}
|
|
|
|
return count;
|
|
|
|
} /* countAddrRecs */
|
|
|
|
|
2014-10-15 07:26:18 -07:00
|
|
|
XP_Bool
|
2014-09-25 20:05:37 -07:00
|
|
|
addr_iter( const CommsAddrRec* addr, CommsConnType* typp, XP_U32* state )
|
2015-07-01 07:10:45 -07:00
|
|
|
{
|
2017-01-07 07:41:38 -08:00
|
|
|
XP_Bool result = types_iter( addr->_conTypes, typp, state );
|
|
|
|
return result;
|
2015-07-01 07:10:45 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
XP_Bool
|
|
|
|
types_iter( XP_U32 conTypes, CommsConnType* typp, XP_U32* state )
|
2014-09-25 20:05:37 -07:00
|
|
|
{
|
|
|
|
CommsConnType typ = *state;
|
2017-01-07 07:41:38 -08:00
|
|
|
XP_ASSERT( typ < COMMS_CONN_NTYPES );
|
2014-09-25 20:05:37 -07:00
|
|
|
while ( ++typ < COMMS_CONN_NTYPES ) {
|
|
|
|
*state = typ;
|
|
|
|
XP_U16 mask = 1 << (typ - 1);
|
2014-09-25 22:24:09 -07:00
|
|
|
if ( mask == (conTypes & mask) ) {
|
2014-09-25 20:05:37 -07:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
2014-09-25 22:24:09 -07:00
|
|
|
XP_Bool found = typ < COMMS_CONN_NTYPES;
|
|
|
|
if ( found ) {
|
2014-09-25 20:05:37 -07:00
|
|
|
*typp = typ;
|
|
|
|
}
|
2014-11-05 07:41:20 -08:00
|
|
|
// XP_LOGF( "%s(flag=%x)=>%d (typ=%s)", __func__, conTypes, found, ConnType2Str( typ ) );
|
2014-09-25 22:24:09 -07:00
|
|
|
return found;
|
2014-09-25 20:05:37 -07:00
|
|
|
}
|
|
|
|
|
2014-10-15 07:26:18 -07:00
|
|
|
XP_Bool
|
2014-11-05 07:41:20 -08:00
|
|
|
addr_hasType( const CommsAddrRec* addr, CommsConnType typ )
|
2015-07-01 07:10:45 -07:00
|
|
|
{
|
|
|
|
return types_hasType( addr->_conTypes, typ );
|
|
|
|
}
|
|
|
|
|
|
|
|
XP_Bool
|
|
|
|
types_hasType( XP_U16 conTypes, CommsConnType typ )
|
2014-10-15 07:26:18 -07:00
|
|
|
{
|
2014-11-05 07:41:20 -08:00
|
|
|
/* Any address has NONE */
|
|
|
|
XP_Bool hasType = COMMS_CONN_NONE == typ;
|
|
|
|
if ( !hasType ) {
|
2015-07-01 07:10:45 -07:00
|
|
|
hasType = 0 != (conTypes & (1 << (typ - 1)));
|
2014-11-05 07:41:20 -08:00
|
|
|
}
|
|
|
|
// XP_LOGF( "%s(%s) => %d", __func__, ConnType2Str(typ), hasType );
|
|
|
|
return hasType;
|
2014-10-15 07:26:18 -07:00
|
|
|
}
|
|
|
|
|
2023-12-18 17:08:51 -08:00
|
|
|
XP_Bool
|
|
|
|
addr_isEmpty( const CommsAddrRec* addr )
|
|
|
|
{
|
|
|
|
CommsConnType typ;
|
|
|
|
XP_U32 st = 0;
|
|
|
|
return !addr_iter( addr, &typ, &st );
|
|
|
|
}
|
|
|
|
|
2014-09-25 20:05:37 -07:00
|
|
|
CommsConnType
|
|
|
|
addr_getType( const CommsAddrRec* addr )
|
|
|
|
{
|
|
|
|
CommsConnType typ;
|
2014-09-25 22:24:09 -07:00
|
|
|
XP_U32 st = 0;
|
|
|
|
if ( !addr_iter( addr, &typ, &st ) ) {
|
2014-09-25 20:05:37 -07:00
|
|
|
typ = COMMS_CONN_NONE;
|
|
|
|
}
|
2014-11-05 07:41:20 -08:00
|
|
|
XP_ASSERT( !addr_iter( addr, &typ, &st ) ); /* shouldn't be a second */
|
2014-11-20 07:12:25 -08:00
|
|
|
// XP_LOGF( "%s(%p) => %s", __func__, addr, ConnType2Str( typ ) );
|
2014-09-25 20:05:37 -07:00
|
|
|
return typ;
|
|
|
|
}
|
|
|
|
|
2014-10-15 07:26:18 -07:00
|
|
|
void
|
2015-07-01 07:10:45 -07:00
|
|
|
types_addType( XP_U16* conTypes, CommsConnType type )
|
2014-10-15 07:26:18 -07:00
|
|
|
{
|
|
|
|
XP_ASSERT( COMMS_CONN_NONE != type );
|
2014-11-20 07:12:25 -08:00
|
|
|
// XP_LOGF( "%s(%s)", __func__, ConnType2Str(type) );
|
2015-07-01 07:10:45 -07:00
|
|
|
*conTypes |= 1 << (type - 1);
|
|
|
|
}
|
|
|
|
|
2022-09-03 18:47:04 -07:00
|
|
|
void
|
|
|
|
types_rmType( XP_U16* conTypes, CommsConnType type )
|
|
|
|
{
|
|
|
|
XP_ASSERT( COMMS_CONN_NONE != type );
|
|
|
|
*conTypes &= ~(1 << (type - 1));
|
|
|
|
}
|
|
|
|
|
2015-07-01 07:10:45 -07:00
|
|
|
void
|
|
|
|
addr_addType( CommsAddrRec* addr, CommsConnType type )
|
|
|
|
{
|
|
|
|
types_addType( &addr->_conTypes, type );
|
2014-10-15 07:26:18 -07:00
|
|
|
}
|
|
|
|
|
2014-10-28 08:03:19 -07:00
|
|
|
void
|
|
|
|
addr_rmType( CommsAddrRec* addr, CommsConnType type )
|
|
|
|
{
|
|
|
|
XP_ASSERT( COMMS_CONN_NONE != type );
|
2014-11-20 07:12:25 -08:00
|
|
|
// XP_LOGF( "%s(%s)", __func__, ConnType2Str(type) );
|
2022-09-03 18:47:04 -07:00
|
|
|
types_rmType( &addr->_conTypes, type );
|
2014-10-28 08:03:19 -07:00
|
|
|
}
|
|
|
|
|
2017-01-07 07:41:38 -08:00
|
|
|
/* Overwrites anything that might already be there. Use addr_addType() to add
|
|
|
|
to the set */
|
2014-09-25 20:05:37 -07:00
|
|
|
void
|
|
|
|
addr_setType( CommsAddrRec* addr, CommsConnType type )
|
|
|
|
{
|
2022-04-29 09:19:44 -07:00
|
|
|
XP_LOGFF( "(%p, %s)", addr, ConnType2Str(type) );
|
2014-09-25 20:05:37 -07:00
|
|
|
XP_U16 flags = 0;
|
|
|
|
if ( COMMS_CONN_NONE != type ) {
|
|
|
|
flags = 1 << (type - 1);
|
|
|
|
}
|
|
|
|
addr->_conTypes = flags;
|
|
|
|
XP_ASSERT( type == addr_getType( addr ) );
|
|
|
|
}
|
|
|
|
|
2006-10-10 01:34:37 +00:00
|
|
|
#ifdef XWFEATURE_RELAY
|
2006-06-16 01:46:47 +00:00
|
|
|
static XWHostID
|
2020-04-26 13:49:54 -07:00
|
|
|
getDestID( CommsCtxt* comms, XWEnv XP_UNUSED_DBG(xwe), XP_PlayerAddr channelNo )
|
2006-06-16 01:46:47 +00:00
|
|
|
{
|
|
|
|
XWHostID id = HOST_ID_NONE;
|
2014-12-05 06:19:02 -08:00
|
|
|
XP_Bool missingRelay = XP_FALSE;
|
2009-09-26 14:37:49 +00:00
|
|
|
if ( (channelNo & CHANNEL_MASK) == CHANNEL_NONE ) {
|
2006-06-16 01:46:47 +00:00
|
|
|
id = HOST_ID_SERVER;
|
|
|
|
} else {
|
2014-12-05 06:19:02 -08:00
|
|
|
XP_PlayerAddr masked = channelNo & ~CHANNEL_MASK;
|
2014-11-05 21:57:25 -08:00
|
|
|
for ( AddressRecord* recs = comms->recs; !!recs; recs = recs->next ) {
|
2014-11-06 20:28:51 -08:00
|
|
|
CNO_FMT( cbuf, recs->channelNo );
|
2022-04-29 09:19:44 -07:00
|
|
|
XP_LOGFF( "rec %p has %s, hostID %d", recs,
|
2014-11-06 20:28:51 -08:00
|
|
|
cbuf, recs->rr.hostID );
|
2014-12-05 06:19:02 -08:00
|
|
|
if ( (recs->channelNo & ~CHANNEL_MASK) != masked ) {
|
2022-04-29 09:19:44 -07:00
|
|
|
XP_LOGFF( "rejecting record %p; channelNo doesn't match", recs );
|
2023-02-03 22:00:47 -08:00
|
|
|
logAddr( comms, &recs->addr, __func__ );
|
2014-12-05 06:19:02 -08:00
|
|
|
} else if ( !addr_hasType( &recs->addr, COMMS_CONN_RELAY ) ) {
|
2022-04-29 09:19:44 -07:00
|
|
|
XP_LOGFF( "rejecting record %p; no relay address", recs );
|
2023-02-03 22:00:47 -08:00
|
|
|
logAddr( comms, &recs->addr, __func__ );
|
2014-12-05 06:19:02 -08:00
|
|
|
missingRelay = XP_TRUE;
|
|
|
|
} else {
|
2014-12-10 07:54:50 -08:00
|
|
|
XP_ASSERT( HOST_ID_NONE == id ); /* no duplicates */
|
2014-12-05 06:19:02 -08:00
|
|
|
id = recs->rr.hostID;
|
2014-12-10 07:54:50 -08:00
|
|
|
// break;
|
2006-06-16 01:46:47 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2014-12-05 06:19:02 -08:00
|
|
|
|
|
|
|
/* If we get here AND we're a client, it may be the case that the server
|
|
|
|
channel is what we want because though we haven't connected via relay
|
|
|
|
yet we have a channel working via another transport. */
|
|
|
|
if ( HOST_ID_NONE == id && missingRelay && !comms->isServer ) {
|
2022-04-29 09:19:44 -07:00
|
|
|
XP_LOGFF( "special casing channel missing relay address" );
|
2014-12-05 06:19:02 -08:00
|
|
|
id = HOST_ID_SERVER;
|
|
|
|
}
|
|
|
|
|
2014-11-06 18:44:19 -08:00
|
|
|
CNO_FMT( cbuf, channelNo );
|
2022-04-29 09:19:44 -07:00
|
|
|
XP_LOGFF( "(%s) => %x", cbuf, id );
|
2006-06-16 01:46:47 +00:00
|
|
|
return id;
|
|
|
|
} /* getDestID */
|
|
|
|
|
2011-08-16 19:36:23 -07:00
|
|
|
static XWStreamCtxt*
|
2020-04-25 17:47:07 -07:00
|
|
|
relay_msg_to_stream( CommsCtxt* comms, XWEnv xwe, XWRELAY_Cmd cmd, XWHostID destID,
|
2014-12-05 06:19:02 -08:00
|
|
|
void* data, int datalen )
|
2005-03-19 22:01:38 +00:00
|
|
|
{
|
2022-04-29 09:19:44 -07:00
|
|
|
XP_LOGFF( "(cmd=%s, destID=%x)", relayCmdToStr(cmd), destID );
|
2011-08-16 19:36:23 -07:00
|
|
|
XWStreamCtxt* stream;
|
2018-06-27 23:42:50 -07:00
|
|
|
stream = mem_stream_make_raw( MPPARM(comms->mpool)
|
2018-07-05 07:58:50 -07:00
|
|
|
dutil_getVTManager(comms->dutil) );
|
2011-08-16 19:36:23 -07:00
|
|
|
if ( stream != NULL ) {
|
|
|
|
CommsAddrRec addr;
|
|
|
|
stream_putU8( stream, cmd );
|
2005-03-19 22:01:38 +00:00
|
|
|
|
2011-08-16 19:36:23 -07:00
|
|
|
comms_getAddr( comms, &addr );
|
2005-03-19 22:01:38 +00:00
|
|
|
|
2005-07-06 00:58:20 +00:00
|
|
|
switch ( cmd ) {
|
|
|
|
case XWRELAY_MSG_TORELAY:
|
2014-03-08 12:47:58 -08:00
|
|
|
if ( COOKIE_ID_NONE == comms->rr.cookieID ) {
|
2022-04-29 09:19:44 -07:00
|
|
|
XP_LOGFF( "cookieID still 0; background send?" );
|
2014-03-08 12:47:58 -08:00
|
|
|
}
|
2014-02-07 21:12:45 -08:00
|
|
|
stream_putU16( stream, comms->rr.cookieID );
|
2011-08-16 19:36:23 -07:00
|
|
|
case XWRELAY_MSG_TORELAY_NOCONN:
|
2014-11-05 21:57:25 -08:00
|
|
|
XP_ASSERT( 0 < comms->rr.myHostID );
|
2014-02-07 21:12:45 -08:00
|
|
|
stream_putU8( stream, comms->rr.myHostID );
|
2014-11-05 21:57:25 -08:00
|
|
|
XP_ASSERT( 0 < destID );
|
2011-08-16 19:36:23 -07:00
|
|
|
stream_putU8( stream, destID );
|
2020-08-24 16:16:20 -07:00
|
|
|
XP_LOGFF( "wrote ids src %d, dest %d", comms->rr.myHostID, destID );
|
2011-08-16 19:36:23 -07:00
|
|
|
if ( data != NULL && datalen > 0 ) {
|
|
|
|
stream_putBytes( stream, data, datalen );
|
2005-03-19 22:01:38 +00:00
|
|
|
}
|
2005-07-06 00:58:20 +00:00
|
|
|
break;
|
2005-09-05 15:33:51 +00:00
|
|
|
case XWRELAY_GAME_CONNECT:
|
2011-08-16 19:36:23 -07:00
|
|
|
stream_putU8( stream, XWRELAY_PROTO_VERSION );
|
2012-01-26 05:43:54 -08:00
|
|
|
stream_putU16( stream, INITIAL_CLIENT_VERS );
|
2011-08-16 19:36:23 -07:00
|
|
|
stringToStream( stream, addr.u.ip_relay.invite );
|
|
|
|
stream_putU8( stream, addr.u.ip_relay.seeksPublicRoom );
|
|
|
|
stream_putU8( stream, addr.u.ip_relay.advertiseRoom );
|
2010-09-10 01:30:40 -07:00
|
|
|
/* XP_ASSERT( cmd == XWRELAY_GAME_RECONNECT */
|
2014-02-07 21:12:45 -08:00
|
|
|
/* || comms->rr.myHostID == HOST_ID_NONE */
|
|
|
|
/* || comms->rr.myHostID == HOST_ID_SERVER ); */
|
2022-04-29 09:19:44 -07:00
|
|
|
XP_LOGFF( "writing nPlayersHere: %d; nPlayersTotal: %d",
|
|
|
|
comms->rr.nPlayersHere, comms->rr.nPlayersTotal );
|
2014-02-07 21:12:45 -08:00
|
|
|
stream_putU8( stream, comms->rr.nPlayersHere );
|
|
|
|
stream_putU8( stream, comms->rr.nPlayersTotal );
|
2011-08-16 19:36:23 -07:00
|
|
|
stream_putU16( stream, comms_getChannelSeed(comms) );
|
|
|
|
stream_putU8( stream, comms->util->gameInfo->dictLang );
|
2020-04-26 11:28:18 -07:00
|
|
|
putDevID( comms, xwe, stream );
|
2015-02-23 18:28:31 -08:00
|
|
|
stream_putU8( stream, comms->forceChannel ); /* "clientIndx" on relay */
|
2014-12-05 06:19:02 -08:00
|
|
|
|
2020-04-25 17:47:07 -07:00
|
|
|
set_relay_state( comms, xwe, COMMS_RELAYSTATE_CONNECT_PENDING );
|
2010-09-10 01:30:40 -07:00
|
|
|
break;
|
|
|
|
|
2005-09-05 15:33:51 +00:00
|
|
|
case XWRELAY_GAME_RECONNECT:
|
2011-08-16 19:36:23 -07:00
|
|
|
stream_putU8( stream, XWRELAY_PROTO_VERSION );
|
2012-01-26 05:43:54 -08:00
|
|
|
stream_putU16( stream, INITIAL_CLIENT_VERS );
|
2011-08-16 19:36:23 -07:00
|
|
|
stringToStream( stream, addr.u.ip_relay.invite );
|
|
|
|
stream_putU8( stream, addr.u.ip_relay.seeksPublicRoom );
|
|
|
|
stream_putU8( stream, addr.u.ip_relay.advertiseRoom );
|
2014-02-07 21:12:45 -08:00
|
|
|
stream_putU8( stream, comms->rr.myHostID );
|
2009-09-26 14:37:49 +00:00
|
|
|
XP_ASSERT( cmd == XWRELAY_GAME_RECONNECT
|
2014-02-07 21:12:45 -08:00
|
|
|
|| comms->rr.myHostID == HOST_ID_NONE
|
|
|
|
|| comms->rr.myHostID == HOST_ID_SERVER );
|
2022-04-29 09:19:44 -07:00
|
|
|
XP_LOGFF( "writing nPlayersHere: %d; nPlayersTotal: %d",
|
|
|
|
comms->rr.nPlayersHere, comms->rr.nPlayersTotal );
|
2014-02-07 21:12:45 -08:00
|
|
|
stream_putU8( stream, comms->rr.nPlayersHere );
|
|
|
|
stream_putU8( stream, comms->rr.nPlayersTotal );
|
2011-08-16 19:36:23 -07:00
|
|
|
stream_putU16( stream, comms_getChannelSeed(comms) );
|
|
|
|
stream_putU8( stream, comms->util->gameInfo->dictLang );
|
2014-02-07 21:12:45 -08:00
|
|
|
stringToStream( stream, comms->rr.connName );
|
2020-04-26 11:28:18 -07:00
|
|
|
putDevID( comms, xwe, stream );
|
2020-04-25 17:47:07 -07:00
|
|
|
set_relay_state( comms, xwe, COMMS_RELAYSTATE_CONNECT_PENDING );
|
2005-09-02 06:26:33 +00:00
|
|
|
break;
|
|
|
|
|
2010-09-14 13:54:52 -07:00
|
|
|
case XWRELAY_ACK:
|
2011-08-16 19:36:23 -07:00
|
|
|
stream_putU8( stream, destID );
|
2010-09-14 13:54:52 -07:00
|
|
|
break;
|
|
|
|
|
2005-09-05 15:33:51 +00:00
|
|
|
case XWRELAY_GAME_DISCONNECT:
|
2014-02-07 21:12:45 -08:00
|
|
|
stream_putU16( stream, comms->rr.cookieID );
|
|
|
|
stream_putU8( stream, comms->rr.myHostID );
|
2005-09-05 15:33:51 +00:00
|
|
|
break;
|
|
|
|
|
2010-09-11 01:23:13 -07:00
|
|
|
#if defined XWFEATURE_RELAY && defined RELAY_HEARTBEAT
|
2005-07-06 00:58:20 +00:00
|
|
|
case XWRELAY_HEARTBEAT:
|
2005-06-23 04:16:53 +00:00
|
|
|
/* Add these for grins. Server can assert they match the IP
|
|
|
|
address it expects 'em on. */
|
2014-02-07 21:12:45 -08:00
|
|
|
stream_putU16( stream, comms->rr.cookieID );
|
|
|
|
stream_putU8( stream, comms->rr.myHostID );
|
2005-07-06 00:58:20 +00:00
|
|
|
break;
|
2007-11-18 23:43:27 +00:00
|
|
|
#endif
|
2005-09-02 06:26:33 +00:00
|
|
|
default:
|
|
|
|
XP_ASSERT(0);
|
2005-03-19 22:01:38 +00:00
|
|
|
}
|
2011-08-16 19:36:23 -07:00
|
|
|
}
|
|
|
|
return stream;
|
2014-12-05 06:19:02 -08:00
|
|
|
} /* relay_msg_to_stream */
|
2011-08-16 19:36:23 -07:00
|
|
|
|
|
|
|
static XP_Bool
|
2020-04-25 17:47:07 -07:00
|
|
|
send_via_relay( CommsCtxt* comms, XWEnv xwe, XWRELAY_Cmd cmd, XWHostID destID,
|
2015-09-30 06:50:04 -07:00
|
|
|
void* data, int dlen, const XP_UCHAR* msgNo )
|
2011-08-16 19:36:23 -07:00
|
|
|
{
|
|
|
|
XP_Bool success = XP_FALSE;
|
2014-12-02 06:52:48 -08:00
|
|
|
if ( comms_getAddrDisabled( comms, COMMS_CONN_RELAY, XP_TRUE ) ) {
|
2022-04-29 09:19:44 -07:00
|
|
|
XP_LOGFF( "dropping message because %s disabled",
|
2014-12-02 06:52:48 -08:00
|
|
|
ConnType2Str( COMMS_CONN_RELAY ) );
|
|
|
|
} else {
|
2014-12-05 06:19:02 -08:00
|
|
|
XWStreamCtxt* tmpStream =
|
2020-04-25 17:47:07 -07:00
|
|
|
relay_msg_to_stream( comms, xwe, cmd, destID, data, dlen );
|
2014-12-02 06:52:48 -08:00
|
|
|
|
|
|
|
if ( tmpStream != NULL ) {
|
|
|
|
XP_U16 len = 0;
|
|
|
|
|
|
|
|
len = stream_getSize( tmpStream );
|
|
|
|
if ( 0 < len ) {
|
|
|
|
XP_U16 result;
|
|
|
|
CommsAddrRec addr;
|
|
|
|
|
|
|
|
comms_getAddr( comms, &addr );
|
2022-04-29 09:19:44 -07:00
|
|
|
XP_LOGFF( "passing %d bytes to sendproc", len );
|
2020-04-25 17:47:07 -07:00
|
|
|
result = (*comms->procs.send)( xwe, stream_getPtr(tmpStream), len,
|
2015-09-30 06:50:04 -07:00
|
|
|
msgNo, &addr, COMMS_CONN_RELAY,
|
|
|
|
gameID(comms),
|
2014-12-02 06:52:48 -08:00
|
|
|
comms->procs.closure );
|
|
|
|
success = result == len;
|
|
|
|
if ( success ) {
|
|
|
|
setHeartbeatTimer( comms );
|
|
|
|
}
|
2005-06-23 04:16:53 +00:00
|
|
|
}
|
2023-02-03 22:00:47 -08:00
|
|
|
stream_destroy( tmpStream );
|
2005-03-19 22:01:38 +00:00
|
|
|
}
|
|
|
|
}
|
2005-06-23 04:16:53 +00:00
|
|
|
return success;
|
|
|
|
} /* send_via_relay */
|
2005-03-19 22:01:38 +00:00
|
|
|
|
2011-08-10 18:28:34 -07:00
|
|
|
static XP_Bool
|
2020-04-25 17:47:07 -07:00
|
|
|
sendNoConn( CommsCtxt* comms, XWEnv xwe, const MsgQueueElem* elem, XWHostID destID )
|
2011-08-10 18:28:34 -07:00
|
|
|
{
|
|
|
|
LOG_FUNC();
|
|
|
|
XP_Bool success = XP_FALSE;
|
|
|
|
|
|
|
|
XP_UCHAR relayID[64];
|
|
|
|
XP_U16 len = sizeof(relayID);
|
|
|
|
success = NULL != comms->procs.sendNoConn
|
2011-08-16 19:36:23 -07:00
|
|
|
&& (0 != (comms->xportFlags & COMMS_XPORT_FLAGS_HASNOCONN))
|
|
|
|
&& formatRelayID( comms, destID, relayID, &len );
|
|
|
|
if ( success ) {
|
|
|
|
XWStreamCtxt* stream =
|
2020-04-25 17:47:07 -07:00
|
|
|
relay_msg_to_stream( comms, xwe, XWRELAY_MSG_TORELAY_NOCONN,
|
2023-03-18 15:15:34 -07:00
|
|
|
destID, elem->smp.buf, elem->smp.len );
|
2011-08-16 19:36:23 -07:00
|
|
|
if ( NULL != stream ) {
|
|
|
|
XP_U16 len = stream_getSize( stream );
|
2011-09-22 17:31:57 -07:00
|
|
|
if ( 0 < len ) {
|
2015-09-30 06:50:04 -07:00
|
|
|
XP_UCHAR msgNo[16];
|
2019-03-18 10:55:40 -07:00
|
|
|
formatMsgNo( comms, elem, msgNo, sizeof(msgNo) );
|
2020-04-25 17:47:07 -07:00
|
|
|
success = (*comms->procs.sendNoConn)( xwe, stream_getPtr( stream ),
|
2015-09-30 06:50:04 -07:00
|
|
|
len, msgNo, relayID,
|
2011-08-16 19:36:23 -07:00
|
|
|
comms->procs.closure );
|
|
|
|
}
|
2023-02-03 22:00:47 -08:00
|
|
|
stream_destroy( stream );
|
2011-08-16 19:36:23 -07:00
|
|
|
}
|
|
|
|
}
|
2011-08-10 18:28:34 -07:00
|
|
|
|
2012-11-13 21:18:52 -08:00
|
|
|
LOG_RETURNF( "%s", success?"TRUE":"FALSE" );
|
2011-08-10 18:28:34 -07:00
|
|
|
return success;
|
|
|
|
}
|
|
|
|
|
2005-03-19 22:01:38 +00: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?
|
|
|
|
*/
|
2009-02-01 15:50:58 +00:00
|
|
|
static XP_Bool
|
2020-04-25 17:47:07 -07:00
|
|
|
relayConnect( CommsCtxt* comms, XWEnv xwe )
|
2005-03-19 22:01:38 +00:00
|
|
|
{
|
2007-02-06 05:49:45 +00:00
|
|
|
LOG_FUNC();
|
2017-11-10 21:34:02 -08:00
|
|
|
XP_Bool success = XP_TRUE;
|
2022-09-03 19:14:40 -07:00
|
|
|
if ( addr_hasType( &comms->selfAddr, COMMS_CONN_RELAY ) ) {
|
2017-11-10 21:34:02 -08:00
|
|
|
if ( 0 ) {
|
|
|
|
#ifdef RELAY_VIA_HTTP
|
|
|
|
} else if ( comms->rr.connName[0] ) {
|
2020-04-25 17:47:07 -07:00
|
|
|
set_relay_state( comms, xwe, COMMS_RELAYSTATE_USING_HTTP );
|
2017-11-10 21:34:02 -08:00
|
|
|
} else {
|
|
|
|
CommsAddrRec addr;
|
|
|
|
comms_getAddr( comms, &addr );
|
|
|
|
DevIDType ignored; /* but should it be? */
|
|
|
|
(*comms->procs.requestJoin)( comms->procs.closure,
|
2020-04-26 11:28:18 -07:00
|
|
|
util_getDevID( comms->util, xwe, &ignored ),
|
2017-11-10 21:34:02 -08:00
|
|
|
addr.u.ip_relay.invite, /* room */
|
|
|
|
comms->rr.nPlayersHere,
|
|
|
|
comms->rr.nPlayersTotal,
|
|
|
|
comms_getChannelSeed(comms),
|
|
|
|
comms->util->gameInfo->dictLang );
|
|
|
|
success = XP_FALSE;
|
|
|
|
#else
|
|
|
|
} else if ( !comms->rr.connecting ) {
|
|
|
|
comms->rr.connecting = XP_TRUE;
|
2020-04-25 17:47:07 -07:00
|
|
|
success = send_via_relay( comms, xwe, comms->rr.connName[0]?
|
2017-11-10 21:34:02 -08:00
|
|
|
XWRELAY_GAME_RECONNECT : XWRELAY_GAME_CONNECT,
|
|
|
|
comms->rr.myHostID, NULL, 0, NULL );
|
|
|
|
comms->rr.connecting = XP_FALSE;
|
|
|
|
#endif
|
|
|
|
}
|
2005-04-03 02:49:04 +00:00
|
|
|
}
|
2009-02-01 15:50:58 +00:00
|
|
|
return success;
|
2005-06-23 04:16:53 +00:00
|
|
|
} /* relayConnect */
|
2006-10-10 01:34:37 +00:00
|
|
|
#endif
|
2005-09-05 15:33:51 +00:00
|
|
|
|
2022-10-07 09:23:09 -07:00
|
|
|
#ifdef DEBUG
|
|
|
|
static void
|
|
|
|
listRecs( const CommsCtxt* comms, const char* msg )
|
|
|
|
{
|
|
|
|
XP_LOGFF( "nrecs: %d", countAddrRecs( comms ) );
|
|
|
|
int ii = 0;
|
|
|
|
for ( AddressRecord* rec = comms->recs; !!rec; rec = rec->next ) {
|
|
|
|
CNO_FMT( cbuf, rec->channelNo );
|
|
|
|
XP_LOGFF( "%s: rec[%d]: %s", msg, ii, cbuf );
|
|
|
|
++ii;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2014-10-15 07:26:18 -07:00
|
|
|
#if defined XWFEATURE_IP_DIRECT || defined XWFEATURE_DIRECTIP
|
2007-02-08 02:53:10 +00:00
|
|
|
static XP_S16
|
2020-04-25 17:47:07 -07:00
|
|
|
send_via_bt_or_ip( CommsCtxt* comms, XWEnv xwe, BTIPMsgType msgTyp, XP_PlayerAddr channelNo,
|
2015-09-30 06:50:04 -07:00
|
|
|
CommsConnType typ, void* data, int dlen, const XP_UCHAR* msgNo )
|
2007-02-08 02:53:10 +00:00
|
|
|
{
|
2007-12-05 06:33:37 +00:00
|
|
|
XP_S16 nSent;
|
2007-02-08 02:53:10 +00:00
|
|
|
XP_U8* buf;
|
2007-12-05 06:33:37 +00:00
|
|
|
LOG_FUNC();
|
|
|
|
nSent = -1;
|
2007-02-08 02:53:10 +00:00
|
|
|
buf = XP_MALLOC( comms->mpool, dlen + 1 );
|
|
|
|
if ( !!buf ) {
|
|
|
|
const CommsAddrRec* addr;
|
2023-02-15 08:31:51 -08:00
|
|
|
(void)channelToAddress( comms, channelNo, &addr );
|
2007-02-08 02:53:10 +00:00
|
|
|
|
2014-10-15 07:26:18 -07:00
|
|
|
buf[0] = msgTyp;
|
2007-02-08 02:53:10 +00:00
|
|
|
if ( dlen > 0 ) {
|
|
|
|
XP_MEMCPY( &buf[1], data, dlen );
|
|
|
|
}
|
|
|
|
|
2022-09-06 12:16:29 -07:00
|
|
|
nSent = (*comms->procs.sendMsg)( xwe, buf, dlen+1, msgNo, 0, addr, typ,
|
|
|
|
gameID(comms), comms->procs.closure );
|
2007-02-08 02:53:10 +00:00
|
|
|
XP_FREE( comms->mpool, buf );
|
2007-11-18 23:43:27 +00:00
|
|
|
|
|
|
|
setHeartbeatTimer( comms );
|
2007-02-08 02:53:10 +00:00
|
|
|
}
|
2007-12-05 06:33:37 +00:00
|
|
|
LOG_RETURNF( "%d", nSent );
|
2007-02-08 02:53:10 +00:00
|
|
|
return nSent;
|
2007-11-26 02:58:25 +00:00
|
|
|
} /* send_via_bt_or_ip */
|
2007-02-08 02:53:10 +00:00
|
|
|
|
2006-08-26 21:12:10 +00:00
|
|
|
#endif
|
|
|
|
|
2006-10-10 01:34:37 +00:00
|
|
|
#ifdef XWFEATURE_RELAY
|
2005-09-05 15:33:51 +00:00
|
|
|
static void
|
2020-04-25 17:47:07 -07:00
|
|
|
relayDisconnect( CommsCtxt* comms, XWEnv xwe )
|
2005-09-05 15:33:51 +00:00
|
|
|
{
|
2009-03-08 19:54:50 +00:00
|
|
|
LOG_FUNC();
|
2022-09-03 19:14:40 -07:00
|
|
|
if ( addr_hasType( &comms->selfAddr, COMMS_CONN_RELAY ) ) {
|
2014-02-07 21:12:45 -08:00
|
|
|
if ( comms->rr.relayState > COMMS_RELAYSTATE_CONNECT_PENDING ) {
|
2020-04-25 17:47:07 -07:00
|
|
|
(void)send_via_relay( comms, xwe, XWRELAY_GAME_DISCONNECT, HOST_ID_NONE,
|
2015-09-30 06:50:04 -07:00
|
|
|
NULL, 0, NULL );
|
2006-08-23 04:44:55 +00:00
|
|
|
}
|
2020-04-25 17:47:07 -07:00
|
|
|
set_relay_state( comms, xwe, COMMS_RELAYSTATE_UNCONNECTED );
|
2005-09-05 15:33:51 +00:00
|
|
|
}
|
|
|
|
} /* relayDisconnect */
|
2012-10-30 07:01:47 -07:00
|
|
|
|
|
|
|
#ifdef XWFEATURE_DEVID
|
|
|
|
static void
|
2020-04-26 11:28:18 -07:00
|
|
|
putDevID( const CommsCtxt* comms, XWEnv xwe, XWStreamCtxt* stream )
|
2012-10-30 07:01:47 -07:00
|
|
|
{
|
|
|
|
# if XWRELAY_PROTO_VERSION >= XWRELAY_PROTO_VERSION_CLIENTID
|
|
|
|
DevIDType typ;
|
2020-04-26 11:28:18 -07:00
|
|
|
const XP_UCHAR* devID = dutil_getDevID( comms->dutil, xwe, &typ );
|
2012-11-06 07:07:14 -08:00
|
|
|
XP_ASSERT( ID_TYPE_NONE <= typ && typ < ID_TYPE_NTYPES );
|
2012-10-30 07:01:47 -07:00
|
|
|
stream_putU8( stream, typ );
|
|
|
|
if ( ID_TYPE_NONE != typ ) {
|
|
|
|
stream_catString( stream, devID );
|
|
|
|
stream_putU8( stream, '\0' );
|
|
|
|
}
|
|
|
|
# else
|
|
|
|
XP_ASSERT(0);
|
|
|
|
XP_USE(comms);
|
|
|
|
XP_USE(stream);
|
|
|
|
# endif
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2005-07-23 15:28:15 +00:00
|
|
|
#endif
|
2005-03-19 22:01:38 +00:00
|
|
|
|
2005-01-31 03:31:50 +00:00
|
|
|
EXTERN_C_END
|
|
|
|
|
2003-11-01 05:35:29 +00:00
|
|
|
#endif /* #ifndef XWFEATURE_STANDALONE_ONLY */
|