2020-01-08 19:19:01 +01:00
|
|
|
/* -*- compile-command: "cd ../linux && make MEMDEBUG=TRUE -j3"; -*- */
|
|
|
|
/*
|
|
|
|
* Copyright 2020 by Eric House (xwords@eehouse.org). All rights reserved.
|
|
|
|
*
|
|
|
|
* 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.
|
|
|
|
*/
|
|
|
|
|
2020-05-31 21:56:37 +02:00
|
|
|
#include <endian.h>
|
2020-11-29 20:43:19 +01:00
|
|
|
#include <inttypes.h>
|
2020-05-31 21:56:37 +02:00
|
|
|
|
2020-01-08 19:19:01 +01:00
|
|
|
#include "device.h"
|
|
|
|
#include "comtypes.h"
|
|
|
|
#include "memstream.h"
|
|
|
|
#include "xwstream.h"
|
2020-05-20 22:58:53 +02:00
|
|
|
#include "strutils.h"
|
|
|
|
#include "nli.h"
|
2022-12-16 22:33:36 +01:00
|
|
|
#include "dbgutil.h"
|
2020-01-08 19:19:01 +01:00
|
|
|
|
|
|
|
static XWStreamCtxt*
|
|
|
|
mkStream( XW_DUtilCtxt* dutil )
|
|
|
|
{
|
|
|
|
XWStreamCtxt* stream = mem_stream_make_raw( MPPARM(dutil->mpool)
|
|
|
|
dutil_getVTManager(dutil) );
|
|
|
|
return stream;
|
|
|
|
}
|
|
|
|
|
2020-05-20 22:58:53 +02:00
|
|
|
#ifdef XWFEATURE_DEVICE
|
|
|
|
|
|
|
|
typedef struct _DevCtxt {
|
|
|
|
XP_U16 devCount;
|
|
|
|
} DevCtxt;
|
|
|
|
|
2020-01-08 19:19:01 +01:00
|
|
|
static DevCtxt*
|
2020-04-26 20:28:18 +02:00
|
|
|
load( XW_DUtilCtxt* dutil, XWEnv xwe )
|
2020-01-08 19:19:01 +01:00
|
|
|
{
|
|
|
|
DevCtxt* state = (DevCtxt*)dutil->devCtxt;
|
|
|
|
if ( NULL == state ) {
|
|
|
|
XWStreamCtxt* stream = mkStream( dutil );
|
2021-03-07 03:51:01 +01:00
|
|
|
const XP_UCHAR* keys[] = { KEY_DEVSTATE, NULL };
|
|
|
|
dutil_loadStream( dutil, xwe, keys, stream );
|
2020-01-08 19:19:01 +01:00
|
|
|
|
|
|
|
state = XP_CALLOC( dutil->mpool, sizeof(*state) );
|
|
|
|
dutil->devCtxt = state;
|
|
|
|
|
|
|
|
if ( 0 < stream_getSize( stream ) ) {
|
|
|
|
state->devCount = stream_getU16( stream );
|
|
|
|
++state->devCount; /* for testing until something's there */
|
2020-01-24 18:05:16 +01:00
|
|
|
/* XP_LOGF( "%s(): read devCount: %d", __func__, state->devCount ); */
|
2020-01-08 19:19:01 +01:00
|
|
|
} else {
|
|
|
|
XP_LOGF( "%s(): empty stream!!", __func__ );
|
|
|
|
}
|
2020-04-26 02:47:07 +02:00
|
|
|
stream_destroy( stream, NULL );
|
2020-01-08 19:19:01 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
return state;
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
2020-05-20 22:58:53 +02:00
|
|
|
dvc_store( XW_DUtilCtxt* dutil, XWEnv xwe )
|
2020-01-08 19:19:01 +01:00
|
|
|
{
|
2020-04-26 20:28:18 +02:00
|
|
|
DevCtxt* state = load( dutil, xwe );
|
2020-01-08 19:19:01 +01:00
|
|
|
XWStreamCtxt* stream = mkStream( dutil );
|
|
|
|
stream_putU16( stream, state->devCount );
|
2021-03-07 03:51:01 +01:00
|
|
|
const XP_UCHAR* keys[] = { KEY_DEVSTATE, NULL };
|
|
|
|
dutil_storeStream( dutil, xwe, keys, stream );
|
2020-04-26 02:47:07 +02:00
|
|
|
stream_destroy( stream, NULL );
|
2020-01-08 19:19:01 +01:00
|
|
|
|
|
|
|
XP_FREEP( dutil->mpool, &dutil->devCtxt );
|
|
|
|
}
|
|
|
|
|
|
|
|
#endif
|
2020-05-20 22:58:53 +02:00
|
|
|
|
2020-09-13 23:48:36 +02:00
|
|
|
// #define BOGUS_ALL_SAME_DEVID
|
2020-11-29 20:43:19 +01:00
|
|
|
#define NUM_RUNS 1 /* up this to see how random things look */
|
2020-09-13 23:48:36 +02:00
|
|
|
|
|
|
|
static void
|
|
|
|
getMQTTDevID( XW_DUtilCtxt* dutil, XWEnv xwe, XP_Bool forceNew, MQTTDevID* devID )
|
2020-05-20 22:58:53 +02:00
|
|
|
{
|
2021-03-05 18:03:32 +01:00
|
|
|
const XP_UCHAR* keys[] = { MQTT_DEVID_KEY, NULL };
|
2020-09-13 23:48:36 +02:00
|
|
|
#ifdef BOGUS_ALL_SAME_DEVID
|
|
|
|
XP_USE(forceNew);
|
|
|
|
MQTTDevID bogusID = 0;
|
|
|
|
XP_UCHAR* str = "ABCDEF0123456789";
|
|
|
|
XP_Bool ok = strToMQTTCDevID( str, &bogusID );
|
|
|
|
XP_ASSERT( ok );
|
|
|
|
|
2020-05-20 22:58:53 +02:00
|
|
|
MQTTDevID tmp = 0;
|
2021-03-05 18:03:32 +01:00
|
|
|
XP_U32 len = sizeof(tmp);
|
|
|
|
dutil_loadPtr( dutil, xwe, keys, &tmp, &len );
|
2020-09-13 23:48:36 +02:00
|
|
|
if ( len != sizeof(tmp) || 0 != XP_MEMCMP( &bogusID, &tmp, sizeof(tmp) ) ) {
|
2021-03-05 18:03:32 +01:00
|
|
|
dutil_storePtr( dutil, xwe, keys, &bogusID, sizeof(bogusID) );
|
2020-09-13 23:48:36 +02:00
|
|
|
}
|
|
|
|
*devID = bogusID;
|
|
|
|
|
|
|
|
#else
|
2020-06-19 20:50:44 +02:00
|
|
|
|
2020-09-13 23:48:36 +02:00
|
|
|
MQTTDevID tmp = 0;
|
2021-02-27 00:33:03 +01:00
|
|
|
XP_U32 len = sizeof(tmp);
|
2020-09-13 23:48:36 +02:00
|
|
|
if ( !forceNew ) {
|
2021-03-05 18:03:32 +01:00
|
|
|
dutil_loadPtr( dutil, xwe, keys, &tmp, &len );
|
2020-09-13 23:48:36 +02:00
|
|
|
}
|
|
|
|
|
2021-03-15 15:52:54 +01:00
|
|
|
/* XP_LOGFF( "len: %d; sizeof(tmp): %zu", len, sizeof(tmp) ); */
|
2020-09-13 23:48:36 +02:00
|
|
|
if ( forceNew || len != sizeof(tmp) ) { /* not found, or bogus somehow */
|
2020-11-29 20:43:19 +01:00
|
|
|
int total = 0;
|
|
|
|
for ( int ii = 0; ii < NUM_RUNS; ++ii ) {
|
|
|
|
tmp = XP_RANDOM();
|
|
|
|
tmp <<= 27;
|
|
|
|
tmp ^= XP_RANDOM();
|
|
|
|
tmp <<= 27;
|
|
|
|
tmp ^= XP_RANDOM();
|
|
|
|
|
|
|
|
int count = 0;
|
|
|
|
MQTTDevID tmp2 = tmp;
|
|
|
|
while ( 0 != tmp2 ) {
|
|
|
|
if ( 0 != (1 & tmp2) ) {
|
|
|
|
++count;
|
|
|
|
++total;
|
|
|
|
}
|
|
|
|
tmp2 >>= 1;
|
|
|
|
}
|
|
|
|
XP_LOGFF( "got: %" PRIX64 " (set: %d/%zd)", tmp, count, sizeof(tmp2)*8 );
|
|
|
|
}
|
|
|
|
XP_LOGFF( "average bits set: %d", total / NUM_RUNS );
|
|
|
|
|
2021-03-05 18:03:32 +01:00
|
|
|
dutil_storePtr( dutil, xwe, keys, &tmp, sizeof(tmp) );
|
2020-09-13 23:48:36 +02:00
|
|
|
|
|
|
|
# ifdef DEBUG
|
2020-05-20 22:58:53 +02:00
|
|
|
XP_UCHAR buf[32];
|
|
|
|
formatMQTTDevID( &tmp, buf, VSIZE(buf) );
|
|
|
|
/* This log statement is required by discon_ok2.py!!! (keep in sync) */
|
2020-09-05 23:56:03 +02:00
|
|
|
XP_LOGFF( "generated id: %s; key: %s", buf, MQTT_DEVID_KEY );
|
2020-09-13 23:48:36 +02:00
|
|
|
# endif
|
2020-05-20 22:58:53 +02:00
|
|
|
}
|
|
|
|
*devID = tmp;
|
2020-09-13 23:48:36 +02:00
|
|
|
#endif
|
2021-03-15 15:52:54 +01:00
|
|
|
// LOG_RETURNF( MQTTDevID_FMT " key: %s", *devID, MQTT_DEVID_KEY );
|
2020-09-13 23:48:36 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
dvc_getMQTTDevID( XW_DUtilCtxt* dutil, XWEnv xwe, MQTTDevID* devID )
|
|
|
|
{
|
|
|
|
getMQTTDevID( dutil, xwe, XP_FALSE, devID );
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
dvc_resetMQTTDevID( XW_DUtilCtxt* dutil, XWEnv xwe )
|
|
|
|
{
|
|
|
|
#ifdef BOGUS_ALL_SAME_DEVID
|
|
|
|
XP_LOGFF( "doing nothing" );
|
|
|
|
XP_USE( dutil );
|
|
|
|
XP_USE( xwe );
|
|
|
|
#else
|
|
|
|
MQTTDevID ignored;
|
|
|
|
getMQTTDevID( dutil, xwe, XP_TRUE, &ignored );
|
|
|
|
#endif
|
2020-05-20 22:58:53 +02:00
|
|
|
}
|
|
|
|
|
2022-12-16 22:33:36 +01:00
|
|
|
static XP_UCHAR*
|
|
|
|
appendToStorage( XP_UCHAR* storage, int* offset,
|
|
|
|
const XP_UCHAR* str )
|
|
|
|
{
|
|
|
|
XP_UCHAR* start = &storage[*offset];
|
|
|
|
start[0] = '\0';
|
|
|
|
XP_STRCAT( start, str );
|
|
|
|
*offset += 1 + XP_STRLEN(str);
|
|
|
|
return start;
|
|
|
|
}
|
|
|
|
|
|
|
|
#ifdef DEBUG
|
|
|
|
static void
|
|
|
|
logPtrs( const char* func, int nTopics, char* topics[] )
|
|
|
|
{
|
|
|
|
for ( int ii = 0; ii < nTopics; ++ii ) {
|
|
|
|
XP_LOGFF( "from %s; topics[%d] = %s", func, ii, topics[ii] );
|
|
|
|
}
|
|
|
|
}
|
|
|
|
#else
|
|
|
|
# define logPtrs(func, nTopics, topics)
|
|
|
|
#endif
|
|
|
|
|
|
|
|
void
|
|
|
|
dvc_getMQTTSubTopics( XW_DUtilCtxt* dutil, XWEnv xwe,
|
2022-12-21 19:27:12 +01:00
|
|
|
XP_UCHAR* storage, XP_U16 XP_UNUSED_DBG(storageLen),
|
2022-12-16 22:33:36 +01:00
|
|
|
XP_U16* nTopics, XP_UCHAR* topics[] )
|
|
|
|
{
|
|
|
|
LOG_FUNC();
|
|
|
|
int offset = 0;
|
|
|
|
XP_U16 count = 0;
|
|
|
|
storage[0] = '\0';
|
|
|
|
|
|
|
|
MQTTDevID devid;
|
|
|
|
getMQTTDevID( dutil, xwe, XP_FALSE, &devid );
|
|
|
|
XP_UCHAR buf[64];
|
2022-12-20 04:53:57 +01:00
|
|
|
formatMQTTDevTopic( &devid, buf, VSIZE(buf) );
|
2022-12-16 22:33:36 +01:00
|
|
|
|
2022-12-20 04:53:57 +01:00
|
|
|
#ifdef MQTT_DEV_TOPICS
|
2022-12-16 22:33:36 +01:00
|
|
|
/* First, the main device topic */
|
|
|
|
topics[count++] = appendToStorage( storage, &offset, buf );
|
2022-12-20 04:53:57 +01:00
|
|
|
#endif
|
2022-12-16 22:33:36 +01:00
|
|
|
|
2022-12-19 01:17:05 +01:00
|
|
|
#ifdef MQTT_GAMEID_TOPICS
|
2022-12-16 22:33:36 +01:00
|
|
|
/* Then the pattern that includes gameIDs */
|
2022-12-20 04:53:57 +01:00
|
|
|
XP_UCHAR buf2[64];
|
|
|
|
size_t siz = XP_SNPRINTF( buf2, VSIZE(buf2), "%s/+", buf );
|
|
|
|
XP_ASSERT( siz < VSIZE(buf) );
|
2022-12-21 19:27:12 +01:00
|
|
|
XP_USE(siz);
|
2022-12-20 04:53:57 +01:00
|
|
|
topics[count++] = appendToStorage( storage, &offset, buf2 );
|
2022-12-19 01:17:05 +01:00
|
|
|
#endif
|
2022-12-16 22:33:36 +01:00
|
|
|
|
|
|
|
/* Finally, the control pattern */
|
|
|
|
formatMQTTCtrlTopic( &devid, buf, VSIZE(buf) );
|
|
|
|
topics[count++] = appendToStorage( storage, &offset, buf );
|
|
|
|
|
|
|
|
for ( int ii = 0; ii < count; ++ii ) {
|
|
|
|
XP_LOGFF( "AFTER: got %d: %s", ii, topics[ii] );
|
|
|
|
}
|
|
|
|
|
|
|
|
XP_ASSERT( count <= *nTopics );
|
|
|
|
*nTopics = count;
|
|
|
|
XP_ASSERT( offset < storageLen );
|
|
|
|
|
|
|
|
logPtrs( __func__, *nTopics, topics );
|
|
|
|
|
|
|
|
LOG_RETURN_VOID();
|
|
|
|
}
|
|
|
|
|
2020-05-20 22:58:53 +02:00
|
|
|
typedef enum { CMD_INVITE, CMD_MSG, CMD_DEVGONE, } MQTTCmd;
|
|
|
|
|
2020-09-08 02:59:53 +02:00
|
|
|
// #define PROTO_0 0
|
2020-06-03 18:21:12 +02:00
|
|
|
#define PROTO_1 1 /* moves gameID into "header" relay2 knows about */
|
2022-06-29 21:29:32 +02:00
|
|
|
#define PROTO_2 2 /* adds timestamp to header */
|
2022-12-20 04:53:57 +01:00
|
|
|
#define PROTO_3 3 /* adds multi-message, removes gameID */
|
2022-06-29 21:29:32 +02:00
|
|
|
#ifndef MQTT_USE_PROTO
|
|
|
|
# define MQTT_USE_PROTO PROTO_1
|
|
|
|
#endif
|
2020-05-20 22:58:53 +02:00
|
|
|
|
|
|
|
static void
|
2020-06-03 18:21:12 +02:00
|
|
|
addHeaderGameIDAndCmd( XW_DUtilCtxt* dutil, XWEnv xwe, MQTTCmd cmd,
|
2022-06-29 21:29:32 +02:00
|
|
|
XP_U32 gameID, XP_U32 timestamp, XWStreamCtxt* stream )
|
2020-05-20 22:58:53 +02:00
|
|
|
{
|
2022-06-29 21:29:32 +02:00
|
|
|
stream_putU8( stream, MQTT_USE_PROTO );
|
2020-05-20 22:58:53 +02:00
|
|
|
|
|
|
|
MQTTDevID myID;
|
|
|
|
dvc_getMQTTDevID( dutil, xwe, &myID );
|
2020-05-31 21:56:37 +02:00
|
|
|
myID = htobe64( myID );
|
2020-05-20 22:58:53 +02:00
|
|
|
stream_putBytes( stream, &myID, sizeof(myID) );
|
|
|
|
|
2020-06-03 18:21:12 +02:00
|
|
|
stream_putU32( stream, gameID );
|
2022-06-29 21:29:32 +02:00
|
|
|
if ( PROTO_2 <= MQTT_USE_PROTO ) {
|
2022-12-30 06:23:15 +01:00
|
|
|
XP_ASSERT(0 != timestamp);
|
2022-06-29 21:29:32 +02:00
|
|
|
stream_putU32( stream, timestamp );
|
|
|
|
}
|
2020-05-31 21:56:37 +02:00
|
|
|
|
2020-06-03 18:21:12 +02:00
|
|
|
stream_putU8( stream, cmd );
|
2020-05-31 21:56:37 +02:00
|
|
|
}
|
|
|
|
|
2022-12-20 04:53:57 +01:00
|
|
|
#ifdef MQTT_GAMEID_TOPICS
|
|
|
|
static void
|
|
|
|
addProto3HeaderCmd( XW_DUtilCtxt* dutil, XWEnv xwe, MQTTCmd cmd,
|
|
|
|
XP_U32 timestamp, XWStreamCtxt* stream )
|
|
|
|
{
|
2022-12-30 06:23:15 +01:00
|
|
|
XP_ASSERT( 0 != timestamp );
|
2022-12-20 04:53:57 +01:00
|
|
|
stream_putU8( stream, PROTO_3 );
|
|
|
|
|
|
|
|
MQTTDevID myID;
|
|
|
|
dvc_getMQTTDevID( dutil, xwe, &myID );
|
|
|
|
myID = htobe64( myID );
|
|
|
|
stream_putBytes( stream, &myID, sizeof(myID) );
|
|
|
|
|
|
|
|
stream_putU32( stream, timestamp );
|
|
|
|
|
|
|
|
stream_putU8( stream, cmd );
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2022-12-20 20:55:30 +01:00
|
|
|
static void
|
|
|
|
callProc( MsgAndTopicProc proc, void* closure, const XP_UCHAR* topic, XWStreamCtxt* stream )
|
|
|
|
{
|
|
|
|
(*proc)( closure, topic, stream_getPtr(stream), stream_getSize(stream) );
|
|
|
|
}
|
|
|
|
|
2020-05-31 21:56:37 +02:00
|
|
|
void
|
2022-12-20 18:20:02 +01:00
|
|
|
dvc_makeMQTTInvites( XW_DUtilCtxt* dutil, XWEnv xwe,
|
|
|
|
MsgAndTopicProc proc, void* closure,
|
|
|
|
const MQTTDevID* addressee,
|
|
|
|
const NetLaunchInfo* nli,
|
|
|
|
XP_U32 timestamp )
|
2020-05-31 21:56:37 +02:00
|
|
|
{
|
2022-12-20 18:20:02 +01:00
|
|
|
XP_UCHAR devTopic[64]; /* used by two below */
|
|
|
|
formatMQTTDevTopic( addressee, devTopic, VSIZE(devTopic) );
|
|
|
|
/* Stream format is identical for both topics */
|
|
|
|
XWStreamCtxt* stream = mkStream( dutil );
|
2022-06-29 21:29:32 +02:00
|
|
|
addHeaderGameIDAndCmd( dutil, xwe, CMD_INVITE, nli->gameID,
|
|
|
|
timestamp, stream );
|
2020-05-31 21:56:37 +02:00
|
|
|
nli_saveToStream( nli, stream );
|
2022-12-20 18:20:02 +01:00
|
|
|
|
|
|
|
#ifdef MQTT_DEV_TOPICS
|
2022-12-20 20:55:30 +01:00
|
|
|
callProc( proc, closure, devTopic, stream );
|
2022-12-20 18:20:02 +01:00
|
|
|
#endif
|
|
|
|
|
|
|
|
#ifdef MQTT_GAMEID_TOPICS
|
|
|
|
XP_UCHAR gameTopic[64];
|
|
|
|
size_t siz = XP_SNPRINTF( gameTopic, VSIZE(gameTopic),
|
|
|
|
"%s/%X", devTopic, nli->gameID );
|
|
|
|
XP_ASSERT( siz < VSIZE(gameTopic) );
|
2022-12-21 19:27:12 +01:00
|
|
|
XP_USE(siz);
|
2022-12-20 20:55:30 +01:00
|
|
|
callProc( proc, closure, devTopic, stream );
|
2022-12-20 18:20:02 +01:00
|
|
|
#endif
|
|
|
|
|
|
|
|
stream_destroy( stream, xwe );
|
2020-05-20 22:58:53 +02:00
|
|
|
}
|
|
|
|
|
2022-12-20 04:53:57 +01:00
|
|
|
/* Ship with == 1, but increase to test */
|
|
|
|
#define MULTI_MSG_COUNT 1
|
|
|
|
|
2020-05-20 22:58:53 +02:00
|
|
|
void
|
2022-12-20 04:53:57 +01:00
|
|
|
dvc_makeMQTTMessages( XW_DUtilCtxt* dutil, XWEnv xwe,
|
|
|
|
MsgAndTopicProc proc, void* closure,
|
|
|
|
const MQTTDevID* addressee,
|
|
|
|
XP_U32 gameID, XP_U32 timestamp,
|
|
|
|
const XP_U8* buf, XP_U16 len )
|
2020-05-20 22:58:53 +02:00
|
|
|
{
|
2022-12-20 04:53:57 +01:00
|
|
|
XP_UCHAR devTopic[64]; /* used by two below */
|
|
|
|
formatMQTTDevTopic( addressee, devTopic, VSIZE(devTopic) );
|
|
|
|
|
|
|
|
#ifdef MQTT_DEV_TOPICS
|
|
|
|
{
|
|
|
|
XWStreamCtxt* stream = mkStream( dutil );
|
|
|
|
addHeaderGameIDAndCmd( dutil, xwe, CMD_MSG, gameID, timestamp, stream );
|
|
|
|
if ( PROTO_2 <= MQTT_USE_PROTO ) {
|
|
|
|
stream_putU32VL( stream, len );
|
|
|
|
}
|
|
|
|
stream_putBytes( stream, buf, len );
|
2022-12-20 20:55:30 +01:00
|
|
|
callProc( proc, closure, devTopic, stream );
|
2022-12-20 04:53:57 +01:00
|
|
|
stream_destroy( stream, xwe );
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#ifdef MQTT_GAMEID_TOPICS
|
|
|
|
{
|
|
|
|
XWStreamCtxt* stream = mkStream( dutil );
|
|
|
|
addProto3HeaderCmd( dutil, xwe, CMD_MSG, timestamp, stream );
|
|
|
|
|
|
|
|
/* For now, we ship one message per packet. But the receiving code
|
|
|
|
should be ready */
|
|
|
|
stream_putU8( stream, MULTI_MSG_COUNT );
|
|
|
|
for ( int ii = 0; ii < MULTI_MSG_COUNT; ++ii ) {
|
|
|
|
stream_putU32VL( stream, len );
|
|
|
|
stream_putBytes( stream, buf, len );
|
|
|
|
}
|
|
|
|
|
|
|
|
XP_UCHAR gameTopic[64];
|
|
|
|
size_t siz = XP_SNPRINTF( gameTopic, VSIZE(gameTopic),
|
|
|
|
"%s/%X", devTopic, gameID );
|
|
|
|
XP_ASSERT( siz < VSIZE(gameTopic) );
|
2022-12-21 19:27:12 +01:00
|
|
|
XP_USE(siz);
|
2022-12-20 04:53:57 +01:00
|
|
|
|
2022-12-20 20:55:30 +01:00
|
|
|
callProc( proc, closure, gameTopic, stream );
|
2022-12-20 04:53:57 +01:00
|
|
|
stream_destroy( stream, xwe );
|
2022-06-29 21:29:32 +02:00
|
|
|
}
|
2022-12-20 04:53:57 +01:00
|
|
|
#endif
|
2020-05-20 22:58:53 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
void
|
2022-12-20 20:55:30 +01:00
|
|
|
dvc_makeMQTTNoSuchGames( XW_DUtilCtxt* dutil, XWEnv xwe,
|
|
|
|
MsgAndTopicProc proc, void* closure,
|
|
|
|
const MQTTDevID* addressee,
|
|
|
|
XP_U32 gameID, XP_U32 timestamp )
|
2020-05-20 22:58:53 +02:00
|
|
|
{
|
2022-12-20 20:55:30 +01:00
|
|
|
XP_UCHAR devTopic[64]; /* used by two below */
|
|
|
|
formatMQTTDevTopic( addressee, devTopic, VSIZE(devTopic) );
|
|
|
|
|
2022-12-30 07:39:02 +01:00
|
|
|
if ( 0 == timestamp ) {
|
|
|
|
timestamp = dutil_getCurSeconds( dutil, xwe );
|
|
|
|
XP_LOGFF( "replacing timestamp of 0" );
|
|
|
|
}
|
|
|
|
|
2022-12-20 20:55:30 +01:00
|
|
|
XWStreamCtxt* stream = mkStream( dutil );
|
2022-06-29 21:29:32 +02:00
|
|
|
addHeaderGameIDAndCmd( dutil, xwe, CMD_DEVGONE, gameID,
|
|
|
|
timestamp, stream );
|
2022-12-20 20:55:30 +01:00
|
|
|
#ifdef MQTT_DEV_TOPICS
|
|
|
|
callProc( proc, closure, devTopic, stream );
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#ifdef MQTT_GAMEID_TOPICS
|
|
|
|
XP_UCHAR gameTopic[64];
|
|
|
|
size_t siz = XP_SNPRINTF( gameTopic, VSIZE(gameTopic),
|
|
|
|
"%s/%X", devTopic, gameID );
|
|
|
|
XP_ASSERT( siz < VSIZE(gameTopic) );
|
2022-12-21 19:27:12 +01:00
|
|
|
XP_USE(siz);
|
2022-12-20 20:55:30 +01:00
|
|
|
callProc( proc, closure, gameTopic, stream );
|
|
|
|
#endif
|
|
|
|
|
|
|
|
stream_destroy( stream, xwe );
|
2020-05-20 22:58:53 +02:00
|
|
|
}
|
|
|
|
|
2022-12-16 22:33:36 +01:00
|
|
|
static XP_Bool
|
2022-12-20 04:53:57 +01:00
|
|
|
isDevMsg( const MQTTDevID* myID, const XP_UCHAR* topic, XP_U32* gameID )
|
2022-12-16 22:33:36 +01:00
|
|
|
{
|
|
|
|
XP_UCHAR buf[64];
|
|
|
|
formatMQTTDevTopic( myID, buf, VSIZE(buf) );
|
2022-12-20 04:53:57 +01:00
|
|
|
size_t topicLen = XP_STRLEN(buf);
|
|
|
|
XP_Bool success = 0 == strncmp( buf, topic, topicLen );
|
|
|
|
if ( success ) {
|
|
|
|
const XP_UCHAR* gameIDPart = topic + topicLen;
|
|
|
|
sscanf( gameIDPart, "/%X", gameID );
|
|
|
|
}
|
|
|
|
// XP_LOGFF( "(%s) => %s (gameID=%X)", topic, boolToStr(success), *gameID );
|
2022-12-16 22:33:36 +01:00
|
|
|
return success;
|
|
|
|
}
|
|
|
|
|
|
|
|
static XP_Bool
|
|
|
|
isCtrlMsg( const MQTTDevID* myID, const XP_UCHAR* topic )
|
|
|
|
{
|
|
|
|
XP_UCHAR buf[64];
|
|
|
|
formatMQTTCtrlTopic( myID, buf, VSIZE(buf) );
|
|
|
|
XP_Bool success = 0 == strncmp( buf, topic, XP_STRLEN(buf) );
|
|
|
|
XP_LOGFF( "(%s) => %s", topic, boolToStr(success) );
|
|
|
|
return success;
|
|
|
|
}
|
|
|
|
|
2022-12-20 04:53:57 +01:00
|
|
|
static void
|
|
|
|
dispatchMsgs( XW_DUtilCtxt* dutil, XWEnv xwe, XP_U8 proto, XWStreamCtxt* stream,
|
|
|
|
XP_U32 gameID, const CommsAddrRec* from )
|
|
|
|
{
|
|
|
|
int msgCount = proto >= PROTO_3 ? stream_getU8( stream ) : 1;
|
|
|
|
for ( int ii = 0; ii < msgCount; ++ii ) {
|
|
|
|
XP_U32 msgLen;
|
|
|
|
if ( PROTO_1 == proto ) {
|
|
|
|
msgLen = stream_getSize( stream );
|
|
|
|
} else {
|
|
|
|
msgLen = stream_getU32VL( stream );
|
|
|
|
}
|
|
|
|
if ( msgLen > stream_getSize( stream ) ) {
|
|
|
|
XP_LOGFF( "msglen %d too large", msgLen );
|
|
|
|
msgLen = 0;
|
|
|
|
XP_ASSERT(0);
|
|
|
|
}
|
|
|
|
if ( 0 < msgLen ) {
|
|
|
|
XP_U8 msgBuf[msgLen];
|
|
|
|
stream_getBytes( stream, msgBuf, msgLen );
|
|
|
|
dutil_onMessageReceived( dutil, xwe, gameID,
|
|
|
|
from, msgBuf, msgLen );
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-05-20 22:58:53 +02:00
|
|
|
void
|
2022-12-16 22:33:36 +01:00
|
|
|
dvc_parseMQTTPacket( XW_DUtilCtxt* dutil, XWEnv xwe, const XP_UCHAR* topic,
|
2022-06-29 21:29:32 +02:00
|
|
|
const XP_U8* buf, XP_U16 len )
|
2020-05-20 22:58:53 +02:00
|
|
|
{
|
2022-12-29 08:19:52 +01:00
|
|
|
XP_LOGFF( "(topic=%s, len=%d)", topic, len );
|
2022-12-16 22:33:36 +01:00
|
|
|
|
|
|
|
MQTTDevID myID;
|
|
|
|
dvc_getMQTTDevID( dutil, xwe, &myID );
|
|
|
|
|
2022-12-20 04:53:57 +01:00
|
|
|
XP_U32 gameID = 0;
|
|
|
|
if ( isDevMsg( &myID, topic, &gameID ) ) {
|
2022-12-16 22:33:36 +01:00
|
|
|
XWStreamCtxt* stream = mkStream( dutil );
|
|
|
|
stream_putBytes( stream, buf, len );
|
2020-05-20 22:58:53 +02:00
|
|
|
|
2022-12-29 08:19:52 +01:00
|
|
|
XP_U8 proto = 0;
|
|
|
|
if ( stream_gotU8( stream, &proto )
|
|
|
|
&& (proto == PROTO_1 || proto == PROTO_2 || proto == PROTO_3 ) ) {
|
2022-12-16 22:33:36 +01:00
|
|
|
MQTTDevID senderID;
|
|
|
|
stream_getBytes( stream, &senderID, sizeof(senderID) );
|
|
|
|
senderID = be64toh( senderID );
|
2021-04-19 16:04:58 +02:00
|
|
|
#ifdef DEBUG
|
2022-12-16 22:33:36 +01:00
|
|
|
XP_UCHAR tmp[32];
|
|
|
|
formatMQTTDevID( &senderID, tmp, VSIZE(tmp) );
|
|
|
|
XP_LOGFF( "senderID: %s", tmp );
|
2021-04-19 16:04:58 +02:00
|
|
|
#endif
|
2022-12-20 04:53:57 +01:00
|
|
|
if ( proto < PROTO_3 ) {
|
|
|
|
gameID = stream_getU32( stream );
|
|
|
|
} else {
|
|
|
|
XP_ASSERT( 0 != gameID );
|
|
|
|
}
|
2020-06-03 18:21:12 +02:00
|
|
|
|
2022-12-16 22:33:36 +01:00
|
|
|
XP_U32 timestamp = 0;
|
2022-12-20 04:53:57 +01:00
|
|
|
if ( PROTO_2 <= proto ) {
|
2022-12-16 22:33:36 +01:00
|
|
|
timestamp = stream_getU32( stream );
|
2022-06-29 21:29:32 +02:00
|
|
|
#ifdef DEBUG
|
2022-12-16 22:33:36 +01:00
|
|
|
if ( 0 < timestamp ) {
|
|
|
|
XP_U32 now = dutil_getCurSeconds( dutil, xwe );
|
|
|
|
XP_LOGFF( "delivery took %ds", now - timestamp );
|
|
|
|
}
|
2022-09-28 16:24:45 +02:00
|
|
|
#else
|
2022-12-16 22:33:36 +01:00
|
|
|
XP_USE( timestamp );
|
2022-06-29 21:29:32 +02:00
|
|
|
#endif
|
2022-12-16 22:33:36 +01:00
|
|
|
}
|
|
|
|
MQTTCmd cmd = stream_getU8( stream );
|
2020-06-03 18:21:12 +02:00
|
|
|
|
2022-12-16 22:33:36 +01:00
|
|
|
/* Need to ack even if discarded/malformed */
|
|
|
|
dutil_ackMQTTMsg( dutil, xwe, gameID, &senderID, buf, len );
|
2021-04-19 16:04:58 +02:00
|
|
|
|
2022-12-16 22:33:36 +01:00
|
|
|
switch ( cmd ) {
|
|
|
|
case CMD_INVITE: {
|
|
|
|
NetLaunchInfo nli = {0};
|
|
|
|
if ( nli_makeFromStream( &nli, stream ) ) {
|
|
|
|
dutil_onInviteReceived( dutil, xwe, &nli );
|
|
|
|
}
|
2020-05-31 21:56:37 +02:00
|
|
|
}
|
2022-12-16 22:33:36 +01:00
|
|
|
break;
|
|
|
|
case CMD_DEVGONE:
|
|
|
|
case CMD_MSG: {
|
|
|
|
CommsAddrRec from = {0};
|
|
|
|
addr_addType( &from, COMMS_CONN_MQTT );
|
|
|
|
from.u.mqtt.devID = senderID;
|
|
|
|
if ( CMD_MSG == cmd ) {
|
2022-12-20 04:53:57 +01:00
|
|
|
dispatchMsgs( dutil, xwe, proto, stream, gameID, &from );
|
2022-12-16 22:33:36 +01:00
|
|
|
} else if ( CMD_DEVGONE == cmd ) {
|
|
|
|
dutil_onGameGoneReceived( dutil, xwe, gameID, &from );
|
2022-06-29 21:29:32 +02:00
|
|
|
}
|
2020-05-31 21:56:37 +02:00
|
|
|
}
|
2022-12-16 22:33:36 +01:00
|
|
|
break;
|
|
|
|
default:
|
|
|
|
XP_LOGFF( "unknown command %d; dropping message", cmd );
|
|
|
|
XP_ASSERT(0);
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
XP_LOGFF( "bad proto %d; dropping packet", proto );
|
2020-05-31 21:56:37 +02:00
|
|
|
}
|
2022-12-16 22:33:36 +01:00
|
|
|
stream_destroy( stream, xwe );
|
|
|
|
} else if ( isCtrlMsg( &myID, topic ) ) {
|
|
|
|
dutil_onCtrlReceived( dutil, xwe, buf, len );
|
2020-05-20 22:58:53 +02:00
|
|
|
}
|
|
|
|
}
|