mirror of
git://xwords.git.sourceforge.net/gitroot/xwords/xwords
synced 2025-01-10 05:26:10 +01:00
126172220f
especially on the 'sms' side it was broken. Now Android's broken for sure.
1026 lines
36 KiB
C
1026 lines
36 KiB
C
/* -*- compile-command: "cd ../linux && make MEMDEBUG=TRUE -j3"; -*- */
|
|
/*
|
|
* Copyright 2018 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.
|
|
*/
|
|
|
|
#include <unistd.h>
|
|
#include <pthread.h>
|
|
|
|
#include "util.h"
|
|
#include "smsproto.h"
|
|
#include "comtypes.h"
|
|
#include "strutils.h"
|
|
|
|
#define MAX_WAIT 3
|
|
// # define MAX_MSG_LEN 50 /* for testing */
|
|
#define MAX_LEN_BINARY 115
|
|
/* PENDING: Might want to make SEND_NOW_SIZE smaller; might as well send now
|
|
if even the smallest new message is likely to put us over. */
|
|
#define SEND_NOW_SIZE MAX_LEN_BINARY
|
|
|
|
/* To match the SMSService format */
|
|
#define SMS_PROTO_VERSION_JAVA 1
|
|
#define SMS_PROTO_VERSION_COMBO 2
|
|
|
|
#define PARTIALS_FORMAT 0
|
|
|
|
typedef struct _MsgRec {
|
|
XP_U32 createSeconds;
|
|
SMSMsgNet msgNet;
|
|
} MsgRec;
|
|
|
|
typedef struct _ToPhoneRec {
|
|
XP_UCHAR phone[32];
|
|
XP_U32 createSeconds;
|
|
XP_U16 nMsgs;
|
|
XP_U16 totalSize;
|
|
MsgRec** msgs;
|
|
} ToPhoneRec;
|
|
|
|
typedef struct _MsgIDRec {
|
|
int msgID;
|
|
int count;
|
|
struct {
|
|
XP_U16 len;
|
|
XP_U8* data;
|
|
}* parts;
|
|
} MsgIDRec;
|
|
|
|
typedef struct _FromPhoneRec {
|
|
XP_UCHAR phone[32];
|
|
int nMsgIDs;
|
|
MsgIDRec* msgIDRecs;
|
|
} FromPhoneRec;
|
|
|
|
struct SMSProto {
|
|
XW_DUtilCtxt* dutil;
|
|
pthread_t creator;
|
|
pthread_mutex_t mutex;
|
|
XP_U16 nNextID;
|
|
int lastStoredSize;
|
|
XP_U16 nToPhones;
|
|
ToPhoneRec* toPhoneRecs;
|
|
|
|
int nFromPhones;
|
|
FromPhoneRec* fromPhoneRecs;
|
|
#ifdef DEBUG
|
|
pthread_t starter;
|
|
int nestCount;
|
|
#endif
|
|
MPSLOT;
|
|
};
|
|
|
|
#define KEY_PARTIALS PERSIST_KEY("partials")
|
|
#define KEY_NEXTID PERSIST_KEY("nextID")
|
|
|
|
static int nextMsgID( SMSProto* state );
|
|
static XWStreamCtxt* mkStream( SMSProto* state );
|
|
static SMSMsgArray* toNetMsgs( SMSProto* state, ToPhoneRec* rec, XP_Bool forceOld );
|
|
static ToPhoneRec* getForPhone( SMSProto* state, const XP_UCHAR* phone,
|
|
XP_Bool create );
|
|
static void addToOutRec( SMSProto* state, ToPhoneRec* rec, SMS_CMD cmd,
|
|
XP_U16 port, XP_U32 gameID, const XP_U8* buf,
|
|
XP_U16 buflen, XP_U32 nowSeconds );
|
|
static void addMessage( SMSProto* state, const XP_UCHAR* fromPhone, int msgID,
|
|
int indx, int count, const XP_U8* data, XP_U16 len );
|
|
static SMSMsgArray* completeMsgs( SMSProto* state, SMSMsgArray* arr,
|
|
const XP_UCHAR* fromPhone, XP_U16 wantPort,
|
|
int msgID );
|
|
static void savePartials( SMSProto* state );
|
|
static void restorePartials( SMSProto* state );
|
|
static void rmFromPhoneRec( SMSProto* state, int fromPhoneIndex );
|
|
static void freeMsgIDRec( SMSProto* state, MsgIDRec* rec, int fromPhoneIndex,
|
|
int msgIDIndex );
|
|
static void freeForPhone( SMSProto* state, const XP_UCHAR* phone );
|
|
static void freeMsg( SMSProto* state, MsgRec** msg );
|
|
static void freeRec( SMSProto* state, ToPhoneRec* rec );
|
|
|
|
SMSProto*
|
|
smsproto_init( MPFORMAL XW_DUtilCtxt* dutil )
|
|
{
|
|
SMSProto* state = (SMSProto*)XP_CALLOC( mpool, sizeof(*state) );
|
|
pthread_mutex_init( &state->mutex, NULL );
|
|
state->dutil = dutil;
|
|
MPASSIGN( state->mpool, mpool );
|
|
|
|
XP_U16 siz = sizeof(state->nNextID);
|
|
dutil_loadPtr( state->dutil, KEY_NEXTID, &state->nNextID, &siz );
|
|
XP_LOGF( "%s(): loaded nextMsgID: %d", __func__, state->nNextID );
|
|
|
|
restorePartials( state );
|
|
|
|
return state;
|
|
}
|
|
|
|
void
|
|
smsproto_free( SMSProto* state )
|
|
{
|
|
if ( NULL != state ) {
|
|
XP_ASSERT( state->creator == 0 || state->creator == pthread_self() );
|
|
|
|
for ( XP_U16 ii = 0; ii < state->nToPhones; ++ii ) {
|
|
freeRec( state, &state->toPhoneRecs[ii] );
|
|
}
|
|
XP_FREEP( state->mpool, &state->toPhoneRecs );
|
|
|
|
if ( 0 < state->nFromPhones ) {
|
|
XP_LOGF( "%s(): freeing undelivered partial messages", __func__ );
|
|
}
|
|
while (0 < state->nFromPhones) {
|
|
FromPhoneRec* ffr = &state->fromPhoneRecs[0];
|
|
while ( 0 < ffr->nMsgIDs ) {
|
|
freeMsgIDRec( state, &ffr->msgIDRecs[0], 0, 0 );
|
|
}
|
|
}
|
|
XP_ASSERT( !state->fromPhoneRecs ); /* above nulls this once empty */
|
|
|
|
pthread_mutex_destroy( &state->mutex );
|
|
|
|
XP_FREEP( state->mpool, &state );
|
|
}
|
|
}
|
|
|
|
static void
|
|
headerToStream( XWStreamCtxt* stream, SMS_CMD cmd, XP_U16 port, XP_U32 gameID )
|
|
{
|
|
// XP_LOGF( "%s(cmd: %d; gameID: %d)", __func__, cmd, gameID );
|
|
stream_putU8( stream, SMS_PROTO_VERSION_JAVA );
|
|
stream_putU16( stream, port );
|
|
stream_putU8( stream, cmd );
|
|
switch ( cmd ) {
|
|
case NONE:
|
|
XP_ASSERT(0);
|
|
break;
|
|
case INVITE:
|
|
break;
|
|
default:
|
|
stream_putU32( stream, gameID );
|
|
}
|
|
}
|
|
|
|
static XP_Bool
|
|
headerFromStream( XWStreamCtxt* stream, SMS_CMD* cmd, XP_U16* port, XP_U32* gameID )
|
|
{
|
|
XP_Bool success = XP_FALSE;
|
|
XP_U8 tmp;
|
|
|
|
if ( stream_gotU8( stream, &tmp )
|
|
&& tmp == SMS_PROTO_VERSION_JAVA
|
|
&& stream_gotU16( stream, port )
|
|
&& stream_gotU8( stream, &tmp ) ) {
|
|
*cmd = tmp;
|
|
switch( *cmd ) {
|
|
case INVITE:
|
|
success = XP_TRUE;
|
|
break;
|
|
default:
|
|
success = stream_gotU32( stream, gameID );
|
|
break;
|
|
}
|
|
}
|
|
// XP_LOGF( "%s() => cmd: %d; gameID: %d", __func__, *cmd, *gameID );
|
|
return success;
|
|
}
|
|
|
|
/* Maintain a list of pending messages per phone number. When called and it's
|
|
* been at least some amount of time since we last added something, or at
|
|
* least some longer time since the oldest message was added, return an array
|
|
* of messages ready to send via the device's raw SMS (i.e. respecting its
|
|
* size limits.)
|
|
|
|
* Pass in the current time, as that's easier than keeping an instance of
|
|
* UtilCtxt around.
|
|
*/
|
|
SMSMsgArray*
|
|
smsproto_prepOutbound( SMSProto* state, SMS_CMD cmd, XP_U32 gameID,
|
|
const void* buf, XP_U16 buflen, const XP_UCHAR* toPhone,
|
|
int toPort, XP_Bool forceOld, XP_U16* waitSecsP )
|
|
{
|
|
XP_USE( toPort );
|
|
SMSMsgArray* result = NULL;
|
|
pthread_mutex_lock( &state->mutex );
|
|
|
|
#ifdef DEBUG
|
|
XP_UCHAR* checksum = dutil_md5sum( state->dutil, buf, buflen );
|
|
XP_LOGF( "%s(cmd=%d, gameID=%d): len=%d, sum=%s, toPhone=%s", __func__, cmd,
|
|
gameID, buflen, checksum, toPhone );
|
|
XP_FREEP( state->mpool, &checksum );
|
|
#endif
|
|
|
|
ToPhoneRec* rec = getForPhone( state, toPhone, cmd != NONE );
|
|
|
|
/* First, add the new message (if present) to the array */
|
|
XP_U32 nowSeconds = dutil_getCurSeconds( state->dutil );
|
|
if ( cmd != NONE ) {
|
|
addToOutRec( state, rec, cmd, toPort, gameID, buf, buflen, nowSeconds );
|
|
}
|
|
|
|
/* rec will be non-null if there's something in it */
|
|
XP_Bool doSend = XP_FALSE;
|
|
if ( rec != NULL ) {
|
|
doSend = forceOld
|
|
|| rec->totalSize > SEND_NOW_SIZE
|
|
|| MAX_WAIT <= nowSeconds - rec->createSeconds;
|
|
/* other criteria? */
|
|
}
|
|
|
|
if ( doSend ) {
|
|
result = toNetMsgs( state, rec, forceOld );
|
|
freeForPhone( state, toPhone );
|
|
}
|
|
|
|
XP_U16 waitSecs = 0;
|
|
if ( !result && !!rec && (rec->nMsgs > 0) ) {
|
|
waitSecs = MAX_WAIT - (nowSeconds - rec->createSeconds);
|
|
}
|
|
*waitSecsP = waitSecs;
|
|
|
|
XP_LOGF( "%s() => %p (len=%d, *waitSecs=%d)", __func__, result,
|
|
!!result ? result->nMsgs : 0, *waitSecsP );
|
|
|
|
pthread_mutex_unlock( &state->mutex );
|
|
return result;
|
|
}
|
|
|
|
static SMSMsgArray*
|
|
appendLocMsg( SMSProto* XP_UNUSED_DBG(state), SMSMsgArray* arr, SMSMsgLoc* msg )
|
|
{
|
|
if ( NULL == arr ) {
|
|
arr = XP_CALLOC( state->mpool, sizeof(*arr) );
|
|
arr->format = FORMAT_LOC;
|
|
} else {
|
|
XP_ASSERT( arr->format == FORMAT_LOC );
|
|
}
|
|
|
|
arr->u.msgsLoc = XP_REALLOC( state->mpool, arr->u.msgsLoc,
|
|
(arr->nMsgs + 1) * sizeof(*arr->u.msgsLoc) );
|
|
arr->u.msgsLoc[arr->nMsgs++] = *msg;
|
|
return arr;
|
|
}
|
|
|
|
static SMSMsgArray*
|
|
appendNetMsg( SMSProto* XP_UNUSED_DBG(state), SMSMsgArray* arr, SMSMsgNet* msg )
|
|
{
|
|
if ( NULL == arr ) {
|
|
arr = XP_CALLOC( state->mpool, sizeof(*arr) );
|
|
arr->format = FORMAT_NET;
|
|
} else {
|
|
XP_ASSERT( arr->format == FORMAT_NET );
|
|
}
|
|
|
|
arr->u.msgsNet = XP_REALLOC( state->mpool, arr->u.msgsNet,
|
|
(arr->nMsgs + 1) * sizeof(*arr->u.msgsNet) );
|
|
arr->u.msgsNet[arr->nMsgs++] = *msg;
|
|
return arr;
|
|
}
|
|
|
|
SMSMsgArray*
|
|
smsproto_prepInbound( SMSProto* state, const XP_UCHAR* fromPhone,
|
|
XP_U16 wantPort, const XP_U8* data, XP_U16 len )
|
|
{
|
|
XP_LOGF( "%s(): len=%d, fromPhone=%s", __func__, len, fromPhone );
|
|
SMSMsgArray* result = NULL;
|
|
pthread_mutex_lock( &state->mutex );
|
|
|
|
XWStreamCtxt* stream = mkStream( state );
|
|
stream_putBytes( stream, data, len );
|
|
|
|
XP_U8 proto;
|
|
if ( stream_gotU8( stream, &proto ) ) {
|
|
switch ( proto ) {
|
|
case SMS_PROTO_VERSION_JAVA: {
|
|
XP_U8 msgID, indx, count;
|
|
if ( stream_gotU8( stream, &msgID )
|
|
&& stream_gotU8( stream, &indx )
|
|
&& stream_gotU8( stream, &count )
|
|
&& indx < count ) {
|
|
XP_U16 len = stream_getSize( stream );
|
|
XP_U8 buf[len];
|
|
stream_getBytes( stream, buf, len );
|
|
addMessage( state, fromPhone, msgID, indx, count, buf, len );
|
|
result = completeMsgs( state, result, fromPhone, wantPort, msgID );
|
|
savePartials( state );
|
|
}
|
|
}
|
|
break;
|
|
case SMS_PROTO_VERSION_COMBO: {
|
|
XP_U8 oneLen, msgID;
|
|
while ( stream_gotU8( stream, &oneLen )
|
|
&& stream_gotU8( stream, &msgID ) ) {
|
|
XP_U8 tmp[oneLen];
|
|
stream_getBytes( stream, tmp, oneLen );
|
|
|
|
XWStreamCtxt* msgStream = mkStream( state );
|
|
stream_putBytes( msgStream, tmp, oneLen );
|
|
|
|
XP_U32 gameID;
|
|
XP_U16 port;
|
|
SMS_CMD cmd;
|
|
if ( headerFromStream( msgStream, &cmd, &port, &gameID ) ) {
|
|
XP_U16 msgLen = stream_getSize( msgStream );
|
|
XP_U8 buf[msgLen];
|
|
if ( stream_gotBytes( msgStream, buf, msgLen ) ) {
|
|
if ( port == wantPort ) {
|
|
SMSMsgLoc msg = { .len = msgLen,
|
|
.cmd = cmd,
|
|
.gameID = gameID,
|
|
.data = XP_MALLOC( state->mpool, msgLen ),
|
|
};
|
|
XP_MEMCPY( msg.data, buf, msgLen );
|
|
result = appendLocMsg( state, result, &msg );
|
|
} else {
|
|
XP_LOGF( "%s(): expected port %d, got %d", __func__,
|
|
wantPort, port );
|
|
}
|
|
}
|
|
}
|
|
stream_destroy( msgStream );
|
|
}
|
|
}
|
|
break;
|
|
default:
|
|
/* Don't assert! happens all the time */
|
|
XP_LOGF( "%s(): unexpected proto %d", __func__, proto );
|
|
break;
|
|
}
|
|
}
|
|
|
|
stream_destroy( stream );
|
|
|
|
XP_LOGF( "%s() => %p (len=%d)", __func__, result, (!!result) ? result->nMsgs : 0 );
|
|
|
|
pthread_mutex_unlock( &state->mutex );
|
|
return result;
|
|
}
|
|
|
|
void
|
|
smsproto_freeMsgArray( SMSProto* state, SMSMsgArray* arr )
|
|
{
|
|
pthread_mutex_lock( &state->mutex );
|
|
|
|
for ( int ii = 0; ii < arr->nMsgs; ++ii ) {
|
|
XP_U8** ptr = arr->format == FORMAT_LOC
|
|
? &arr->u.msgsLoc[ii].data : &arr->u.msgsNet[ii].data;
|
|
XP_FREEP( state->mpool, ptr );
|
|
}
|
|
|
|
void** ptr;
|
|
switch( arr->format ) {
|
|
case FORMAT_LOC:
|
|
ptr = (void**)&arr->u.msgsLoc;
|
|
break;
|
|
case FORMAT_NET:
|
|
ptr = (void**)&arr->u.msgsNet;
|
|
break;
|
|
default:
|
|
XP_ASSERT(0);
|
|
ptr = NULL;
|
|
}
|
|
XP_FREEP( state->mpool, ptr );
|
|
XP_FREEP( state->mpool, &arr );
|
|
pthread_mutex_unlock( &state->mutex );
|
|
}
|
|
|
|
static void
|
|
freeMsg( SMSProto* XP_UNUSED_DBG(state), MsgRec** msgp )
|
|
{
|
|
XP_FREEP( state->mpool, &(*msgp)->msgNet.data );
|
|
XP_FREEP( state->mpool, msgp );
|
|
}
|
|
|
|
static void
|
|
freeRec( SMSProto* state, ToPhoneRec* rec )
|
|
{
|
|
for ( XP_U16 jj = 0; jj < rec->nMsgs; ++jj ) {
|
|
freeMsg( state, &rec->msgs[jj] );
|
|
}
|
|
XP_FREEP( state->mpool, &rec->msgs );
|
|
}
|
|
|
|
static ToPhoneRec*
|
|
getForPhone( SMSProto* state, const XP_UCHAR* phone, XP_Bool create )
|
|
{
|
|
ToPhoneRec* rec = NULL;
|
|
for ( XP_U16 ii = 0; !rec && ii < state->nToPhones; ++ii ) {
|
|
if ( 0 == XP_STRCMP( state->toPhoneRecs[ii].phone, phone ) ) {
|
|
rec = &state->toPhoneRecs[ii];
|
|
}
|
|
}
|
|
|
|
if ( !rec && create ) {
|
|
state->toPhoneRecs = XP_REALLOC( state->mpool, state->toPhoneRecs,
|
|
(1 + state->nToPhones) * sizeof(*state->toPhoneRecs) );
|
|
rec = &state->toPhoneRecs[state->nToPhones++];
|
|
XP_MEMSET( rec, 0, sizeof(*rec) );
|
|
XP_STRCAT( rec->phone, phone );
|
|
}
|
|
|
|
return rec;
|
|
}
|
|
|
|
static void
|
|
freeForPhone( SMSProto* state, const XP_UCHAR* phone )
|
|
{
|
|
for ( XP_U16 ii = 0; ii < state->nToPhones; ++ii ) {
|
|
if ( 0 == XP_STRCMP( state->toPhoneRecs[ii].phone, phone ) ) {
|
|
freeRec( state, &state->toPhoneRecs[ii] );
|
|
|
|
XP_U16 nAbove = state->nToPhones - ii - 1;
|
|
XP_ASSERT( nAbove >= 0 );
|
|
if ( nAbove > 0 ) {
|
|
XP_MEMMOVE( &state->toPhoneRecs[ii], &state->toPhoneRecs[ii+1],
|
|
nAbove * sizeof(*state->toPhoneRecs) );
|
|
}
|
|
--state->nToPhones;
|
|
if ( 0 == state->nToPhones ) {
|
|
XP_FREEP( state->mpool, &state->toPhoneRecs );
|
|
} else {
|
|
state->toPhoneRecs = XP_REALLOC( state->mpool, state->toPhoneRecs,
|
|
state->nToPhones * sizeof(*state->toPhoneRecs) );
|
|
}
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
static void
|
|
addToOutRec( SMSProto* state, ToPhoneRec* rec, SMS_CMD cmd,
|
|
XP_U16 port, XP_U32 gameID, const XP_U8* buf, XP_U16 buflen,
|
|
XP_U32 nowSeconds )
|
|
{
|
|
XWStreamCtxt* stream = mkStream( state );
|
|
headerToStream( stream, cmd, port, gameID );
|
|
stream_putBytes( stream, buf, buflen );
|
|
|
|
MsgRec* mRec = XP_CALLOC( state->mpool, sizeof(*rec) );
|
|
XP_U16 len = stream_getSize( stream );
|
|
mRec->msgNet.len = len;
|
|
mRec->msgNet.data = XP_MALLOC( state->mpool, len );
|
|
XP_MEMCPY( mRec->msgNet.data, stream_getPtr(stream), len );
|
|
stream_destroy( stream );
|
|
|
|
mRec->createSeconds = nowSeconds;
|
|
|
|
rec->msgs = XP_REALLOC( state->mpool, rec->msgs, (1 + rec->nMsgs) * sizeof(*rec->msgs) );
|
|
rec->msgs[rec->nMsgs++] = mRec;
|
|
rec->totalSize += len;
|
|
XP_LOGFF( "added msg to %s of len %d; total now %d", rec->phone, len,
|
|
rec->totalSize );
|
|
|
|
if ( rec->nMsgs == 1 ) {
|
|
rec->createSeconds = nowSeconds;
|
|
}
|
|
}
|
|
|
|
static MsgIDRec*
|
|
getMsgIDRec( SMSProto* state, const XP_UCHAR* fromPhone, int msgID,
|
|
XP_Bool addMissing, int* fromPhoneIndex, int* msgIDIndex )
|
|
{
|
|
MsgIDRec* result = NULL;
|
|
|
|
FromPhoneRec* fromPhoneRec = NULL;
|
|
for ( int ii = 0; ii < state->nFromPhones; ++ii ) {
|
|
if ( 0 == XP_STRCMP( state->fromPhoneRecs[ii].phone, fromPhone ) ) {
|
|
fromPhoneRec = &state->fromPhoneRecs[ii];
|
|
*fromPhoneIndex = ii;
|
|
break;
|
|
}
|
|
}
|
|
|
|
// create and add if not found
|
|
if ( NULL == fromPhoneRec && addMissing ) {
|
|
state->fromPhoneRecs =
|
|
XP_REALLOC( state->mpool, state->fromPhoneRecs,
|
|
(state->nFromPhones + 1) * sizeof(*state->fromPhoneRecs) );
|
|
*fromPhoneIndex = state->nFromPhones;
|
|
fromPhoneRec = &state->fromPhoneRecs[state->nFromPhones++];
|
|
XP_MEMSET( fromPhoneRec, 0, sizeof(*fromPhoneRec) );
|
|
XP_STRCAT( fromPhoneRec->phone, fromPhone );
|
|
}
|
|
|
|
// Now find msgID record
|
|
if ( NULL != fromPhoneRec ) {
|
|
for ( int ii = 0; ii < fromPhoneRec->nMsgIDs; ++ii ) {
|
|
if ( fromPhoneRec->msgIDRecs[ii].msgID == msgID ) {
|
|
result = &fromPhoneRec->msgIDRecs[ii];
|
|
*msgIDIndex = ii;
|
|
break;
|
|
}
|
|
}
|
|
|
|
// create and add if not found
|
|
if ( NULL == result && addMissing ) {
|
|
fromPhoneRec->msgIDRecs = XP_REALLOC( state->mpool, fromPhoneRec->msgIDRecs,
|
|
(fromPhoneRec->nMsgIDs + 1)
|
|
* sizeof(*fromPhoneRec->msgIDRecs) );
|
|
MsgIDRec newRec = { .msgID = msgID };
|
|
*msgIDIndex = fromPhoneRec->nMsgIDs;
|
|
result = &fromPhoneRec->msgIDRecs[fromPhoneRec->nMsgIDs];
|
|
fromPhoneRec->msgIDRecs[fromPhoneRec->nMsgIDs++] = newRec;
|
|
}
|
|
}
|
|
|
|
return result;
|
|
}
|
|
|
|
/* Messages that are split gather here until complete
|
|
*/
|
|
static void
|
|
addMessage( SMSProto* state, const XP_UCHAR* fromPhone, int msgID, int indx,
|
|
int count, const XP_U8* data, XP_U16 len )
|
|
{
|
|
XP_LOGFF( "phone=%s, msgID=%d, %d/%d", fromPhone, msgID, indx, count );
|
|
XP_ASSERT( 0 < len );
|
|
MsgIDRec* msgIDRec;
|
|
for ( ; ; ) {
|
|
int fromPhoneIndex;
|
|
int msgIDIndex;
|
|
msgIDRec = getMsgIDRec( state, fromPhone, msgID, XP_TRUE,
|
|
&fromPhoneIndex, &msgIDIndex );
|
|
|
|
/* sanity check... */
|
|
if ( msgIDRec->count == 0 || msgIDRec->count == count ) {
|
|
break;
|
|
}
|
|
freeMsgIDRec( state, msgIDRec, fromPhoneIndex, msgIDIndex );
|
|
}
|
|
|
|
/* if it's new, fill in missing fields */
|
|
if ( msgIDRec->count == 0 ) {
|
|
msgIDRec->count = count; /* in case it's new */
|
|
msgIDRec->parts = XP_CALLOC( state->mpool, count * sizeof(*msgIDRec->parts));
|
|
}
|
|
|
|
XP_ASSERT( msgIDRec->parts[indx].len == 0
|
|
|| msgIDRec->parts[indx].len == len ); /* replace with same ok */
|
|
msgIDRec->parts[indx].len = len;
|
|
XP_FREEP( state->mpool, &msgIDRec->parts[indx].data ); /* in case non-null (replacement) */
|
|
msgIDRec->parts[indx].data = XP_MALLOC( state->mpool, len );
|
|
XP_MEMCPY( msgIDRec->parts[indx].data, data, len );
|
|
}
|
|
|
|
static void
|
|
rmFromPhoneRec( SMSProto* state, int fromPhoneIndex )
|
|
{
|
|
FromPhoneRec* fromPhoneRec = &state->fromPhoneRecs[fromPhoneIndex];
|
|
XP_ASSERT( fromPhoneRec->nMsgIDs == 0 );
|
|
XP_FREEP( state->mpool, &fromPhoneRec->msgIDRecs );
|
|
|
|
if ( --state->nFromPhones == 0 ) {
|
|
XP_FREEP( state->mpool, &state->fromPhoneRecs );
|
|
} else {
|
|
XP_U16 nAbove = state->nFromPhones - fromPhoneIndex;
|
|
XP_ASSERT( nAbove >= 0 );
|
|
if ( nAbove > 0 ) {
|
|
XP_MEMMOVE( &state->fromPhoneRecs[fromPhoneIndex], &state->fromPhoneRecs[fromPhoneIndex+1],
|
|
nAbove * sizeof(*state->fromPhoneRecs) );
|
|
}
|
|
state->fromPhoneRecs = XP_REALLOC( state->mpool, state->fromPhoneRecs,
|
|
state->nFromPhones * sizeof(*state->fromPhoneRecs));
|
|
}
|
|
}
|
|
|
|
static void
|
|
freeMsgIDRec( SMSProto* state, MsgIDRec* XP_UNUSED_DBG(rec), int fromPhoneIndex, int msgIDIndex )
|
|
{
|
|
FromPhoneRec* fromPhoneRec = &state->fromPhoneRecs[fromPhoneIndex];
|
|
MsgIDRec* msgIDRec = &fromPhoneRec->msgIDRecs[msgIDIndex];
|
|
XP_ASSERT( msgIDRec == rec );
|
|
|
|
for ( int ii = 0; ii < msgIDRec->count; ++ii ) {
|
|
XP_FREEP( state->mpool, &msgIDRec->parts[ii].data );
|
|
}
|
|
XP_FREEP( state->mpool, &msgIDRec->parts );
|
|
|
|
if ( --fromPhoneRec->nMsgIDs > 0 ) {
|
|
XP_U16 nAbove = fromPhoneRec->nMsgIDs - msgIDIndex;
|
|
XP_ASSERT( nAbove >= 0 );
|
|
if ( nAbove > 0 ) {
|
|
XP_MEMMOVE( &fromPhoneRec->msgIDRecs[msgIDIndex], &fromPhoneRec->msgIDRecs[msgIDIndex+1],
|
|
nAbove * sizeof(*fromPhoneRec->msgIDRecs) );
|
|
}
|
|
fromPhoneRec->msgIDRecs = XP_REALLOC( state->mpool, fromPhoneRec->msgIDRecs,
|
|
fromPhoneRec->nMsgIDs
|
|
* sizeof(*fromPhoneRec->msgIDRecs));
|
|
} else {
|
|
rmFromPhoneRec( state, fromPhoneIndex );
|
|
}
|
|
}
|
|
|
|
static void
|
|
savePartials( SMSProto* state )
|
|
{
|
|
XWStreamCtxt* stream = mkStream( state );
|
|
stream_putU8( stream, PARTIALS_FORMAT );
|
|
|
|
stream_putU8( stream, state->nFromPhones );
|
|
for ( int ii = 0; ii < state->nFromPhones; ++ii ) {
|
|
const FromPhoneRec* rec = &state->fromPhoneRecs[ii];
|
|
stringToStream( stream, rec->phone );
|
|
stream_putU8( stream, rec->nMsgIDs );
|
|
for ( int jj = 0; jj < rec->nMsgIDs; ++jj ) {
|
|
MsgIDRec* mir = &rec->msgIDRecs[jj];
|
|
stream_putU16( stream, mir->msgID );
|
|
stream_putU8( stream, mir->count );
|
|
|
|
/* There's an array here. It may be sparse. Save a len of 0 */
|
|
for ( int kk = 0; kk < mir->count; ++kk ) {
|
|
int len = mir->parts[kk].len;
|
|
stream_putU8( stream, len );
|
|
stream_putBytes( stream, mir->parts[kk].data, len );
|
|
}
|
|
}
|
|
}
|
|
|
|
XP_U16 newSize = stream_getSize( stream );
|
|
if ( state->lastStoredSize == 2 && newSize == 2 ) {
|
|
XP_LOGFF( "%s", "not storing empty again" );
|
|
} else {
|
|
dutil_storeStream( state->dutil, KEY_PARTIALS, stream );
|
|
state->lastStoredSize = newSize;
|
|
}
|
|
|
|
stream_destroy( stream );
|
|
|
|
LOG_RETURN_VOID();
|
|
} /* savePartials */
|
|
|
|
static void
|
|
restorePartials( SMSProto* state )
|
|
{
|
|
XWStreamCtxt* stream = mkStream( state );
|
|
|
|
dutil_loadStream( state->dutil, KEY_PARTIALS, stream );
|
|
if ( stream_getSize( stream ) >= 1
|
|
&& PARTIALS_FORMAT == stream_getU8( stream ) ) {
|
|
int nFromPhones = stream_getU8( stream );
|
|
for ( int ii = 0; ii < nFromPhones; ++ii ) {
|
|
XP_UCHAR phone[32];
|
|
(void)stringFromStreamHere( stream, phone, VSIZE(phone) );
|
|
int nMsgIDs = stream_getU8( stream );
|
|
/* XP_LOGF( "%s(): got %d message records for phone %s", __func__, */
|
|
/* nMsgIDs, phone ); */
|
|
for ( int jj = 0; jj < nMsgIDs; ++jj ) {
|
|
XP_U16 msgID = stream_getU16( stream );
|
|
int count = stream_getU8( stream );
|
|
/* XP_LOGF( "%s(): got %d records for msgID %d", __func__, count, msgID ); */
|
|
for ( int kk = 0; kk < count; ++kk ) {
|
|
int len = stream_getU8( stream );
|
|
if ( 0 < len ) {
|
|
XP_U8 buf[len];
|
|
stream_getBytes( stream, buf, len );
|
|
addMessage( state, phone, msgID, kk, count, buf, len );
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
stream_destroy( stream );
|
|
}
|
|
|
|
static SMSMsgArray*
|
|
completeMsgs( SMSProto* state, SMSMsgArray* arr, const XP_UCHAR* fromPhone,
|
|
XP_U16 wantPort, int msgID )
|
|
{
|
|
int fromPhoneIndex, msgIDIndex;
|
|
MsgIDRec* rec = getMsgIDRec( state, fromPhone, msgID, XP_FALSE,
|
|
&fromPhoneIndex, &msgIDIndex);
|
|
if ( !rec ) {
|
|
XP_LOGFF( "no rec for phone %s, msgID %d", fromPhone, msgID );
|
|
XP_ASSERT( 0 );
|
|
}
|
|
|
|
int len = 0;
|
|
XP_Bool haveAll = XP_TRUE;
|
|
for ( int ii = 0; ii < rec->count; ++ii ) {
|
|
if ( rec->parts[ii].len == 0 ) {
|
|
haveAll = XP_FALSE;
|
|
break;
|
|
} else {
|
|
len += rec->parts[ii].len;
|
|
}
|
|
}
|
|
|
|
if ( haveAll ) {
|
|
XWStreamCtxt* stream = mkStream( state );
|
|
for ( int ii = 0; ii < rec->count; ++ii ) {
|
|
stream_putBytes( stream, rec->parts[ii].data, rec->parts[ii].len );
|
|
}
|
|
|
|
XP_U32 gameID;
|
|
XP_U16 port;
|
|
SMS_CMD cmd;
|
|
if ( headerFromStream( stream, &cmd, &port, &gameID ) ) {
|
|
XP_U16 len = stream_getSize( stream );
|
|
SMSMsgLoc msg = { .len = len,
|
|
.cmd = cmd,
|
|
.gameID = gameID,
|
|
.data = XP_MALLOC( state->mpool, len ),
|
|
};
|
|
if ( stream_gotBytes( stream, msg.data, len ) && port == wantPort ) {
|
|
arr = appendLocMsg( state, arr, &msg );
|
|
} else {
|
|
XP_LOGFF( "expected port %d, got %d", wantPort, port );
|
|
XP_FREEP( state->mpool, &msg.data );
|
|
}
|
|
}
|
|
stream_destroy( stream );
|
|
|
|
freeMsgIDRec( state, rec, fromPhoneIndex, msgIDIndex );
|
|
}
|
|
|
|
return arr;
|
|
}
|
|
|
|
static SMSMsgArray*
|
|
toNetMsgs( SMSProto* state, ToPhoneRec* rec, XP_Bool forceOld )
|
|
{
|
|
SMSMsgArray* result = NULL;
|
|
|
|
for ( XP_U16 ii = 0; ii < rec->nMsgs; ) {
|
|
// XP_LOGF( "%s(): looking at msg %d of %d", __func__, ii, rec->nMsgs );
|
|
XP_U16 count = (rec->msgs[ii]->msgNet.len + (MAX_LEN_BINARY-1)) / MAX_LEN_BINARY;
|
|
|
|
/* First, see if this message and some number of its neighbors can be
|
|
combined */
|
|
int last = ii;
|
|
int sum = 0;
|
|
if ( count == 1 && !forceOld ) {
|
|
for ( ; last < rec->nMsgs; ++last ) {
|
|
int nextLen = rec->msgs[last]->msgNet.len;
|
|
if ( sum + nextLen > MAX_LEN_BINARY ) {
|
|
break;
|
|
}
|
|
sum += nextLen;
|
|
}
|
|
}
|
|
|
|
if ( last > ii ) {
|
|
int nMsgs = last - ii;
|
|
if ( nMsgs > 1 ) {
|
|
XP_LOGFF( "combining %d through %d (%d msgs)", ii, last - 1, nMsgs );
|
|
}
|
|
int len = 1 + sum + (nMsgs * 2); /* 1: len & msgID */
|
|
SMSMsgNet newMsg = { .len = len,
|
|
.data = XP_MALLOC( state->mpool, len )
|
|
};
|
|
int indx = 0;
|
|
newMsg.data[indx++] = SMS_PROTO_VERSION_COMBO;
|
|
for ( int jj = ii; jj < last; ++jj ) {
|
|
const SMSMsgNet* msg = &rec->msgs[jj]->msgNet;
|
|
newMsg.data[indx++] = msg->len;
|
|
newMsg.data[indx++] = nextMsgID( state );
|
|
XP_MEMCPY( &newMsg.data[indx], msg->data, msg->len ); /* bad! */
|
|
indx += msg->len;
|
|
}
|
|
result = appendNetMsg( state, result, &newMsg );
|
|
ii = last;
|
|
} else {
|
|
int msgID = nextMsgID( state );
|
|
const SMSMsgNet* msg = &rec->msgs[ii]->msgNet;
|
|
XP_U8* nextStart = msg->data;
|
|
XP_U16 lenLeft = msg->len;
|
|
for ( XP_U16 indx = 0; indx < count; ++indx ) {
|
|
XP_ASSERT( lenLeft > 0 );
|
|
XP_U16 useLen = lenLeft;
|
|
if ( useLen >= MAX_LEN_BINARY ) {
|
|
useLen = MAX_LEN_BINARY;
|
|
}
|
|
lenLeft -= useLen;
|
|
|
|
SMSMsgNet newMsg = { .len = useLen + 4,
|
|
.data = XP_MALLOC( state->mpool, useLen + 4 )
|
|
};
|
|
newMsg.data[0] = SMS_PROTO_VERSION_JAVA;
|
|
newMsg.data[1] = msgID;
|
|
newMsg.data[2] = indx;
|
|
newMsg.data[3] = count;
|
|
XP_MEMCPY( newMsg.data + 4, nextStart, useLen );
|
|
nextStart += useLen;
|
|
|
|
result = appendNetMsg( state, result, &newMsg );
|
|
}
|
|
++ii;
|
|
}
|
|
}
|
|
|
|
return result;
|
|
} /* toMsgs */
|
|
|
|
static int
|
|
nextMsgID( SMSProto* state )
|
|
{
|
|
int result = ++state->nNextID % 0x000000FF;
|
|
dutil_storePtr( state->dutil, KEY_NEXTID, &state->nNextID,
|
|
sizeof(state->nNextID) );
|
|
LOG_RETURNF( "%d", result );
|
|
return result;
|
|
}
|
|
|
|
static XWStreamCtxt*
|
|
mkStream( SMSProto* state )
|
|
{
|
|
XWStreamCtxt* stream = mem_stream_make_raw( MPPARM(state->mpool)
|
|
dutil_getVTManager(state->dutil) );
|
|
return stream;
|
|
}
|
|
|
|
#ifdef DEBUG
|
|
void
|
|
smsproto_runTests( MPFORMAL XW_DUtilCtxt* dutil )
|
|
{
|
|
LOG_FUNC();
|
|
SMSProto* state = smsproto_init( mpool, dutil );
|
|
|
|
const int smallSiz = 20;
|
|
const char* phones[] = {"1234", "3456", "5467", "9877"};
|
|
const char* buf = "asoidfaisdfoausdf aiousdfoiu asodfu oiuasdofi oiuaosiduf oaisudf oiasd f"
|
|
";oiaisdjfljiojaklj asdlkjalskdjf laksjd flkjasdlfkj aldsjkf lsakdjf lkjsad flkjsd fl;kj"
|
|
"asdifaoaosidfoiauosidufoaus doifuoaiusdoifu aoisudfoaisd foia sdoifuasodfu aosiud foiuas odfiu asd"
|
|
"aosdoiaosdoiisidfoiosi isoidufoisu doifuoisud oiuoi98a90iu-asjdfoiasdfij"
|
|
;
|
|
const XP_Bool forceOld = XP_TRUE;
|
|
|
|
SMSMsgArray* arrs[VSIZE(phones)];
|
|
for ( int ii = 0; ii < VSIZE(arrs); ++ii ) {
|
|
arrs[ii] = NULL;
|
|
}
|
|
|
|
/* Loop until all the messages are ready. */
|
|
const XP_U32 gameID = 12344321;
|
|
const int port = 1;
|
|
for ( XP_Bool firstTime = XP_TRUE; ; firstTime = XP_FALSE) {
|
|
XP_Bool allDone = XP_TRUE;
|
|
for ( int ii = 0; ii < VSIZE(arrs); ++ii ) {
|
|
XP_U16 waitSecs;
|
|
if ( firstTime ) {
|
|
XP_U16 len = (ii + 1) * 30;
|
|
arrs[ii] = smsproto_prepOutbound( state, DATA, gameID, buf, len, phones[ii],
|
|
port, forceOld, &waitSecs );
|
|
} else if ( NULL == arrs[ii]) {
|
|
arrs[ii] = smsproto_prepOutbound( state, DATA, gameID, NULL, 0, phones[ii],
|
|
port, forceOld, &waitSecs );
|
|
} else {
|
|
continue;
|
|
}
|
|
allDone = allDone & (waitSecs == 0 && !!arrs[ii]);
|
|
}
|
|
if ( allDone ) {
|
|
break;
|
|
} else {
|
|
(void)sleep( 2 );
|
|
}
|
|
}
|
|
|
|
for ( int indx = 0; ; ++indx ) {
|
|
XP_Bool haveOne = XP_FALSE;
|
|
for ( int ii = 0; ii < VSIZE(arrs); ++ii ) {
|
|
if (!!arrs[ii] && indx < arrs[ii]->nMsgs) {
|
|
XP_ASSERT( arrs[ii]->format == FORMAT_NET );
|
|
haveOne = XP_TRUE;
|
|
SMSMsgArray* outArr = smsproto_prepInbound( state, phones[ii], port,
|
|
arrs[ii]->u.msgsNet[indx].data,
|
|
arrs[ii]->u.msgsNet[indx].len );
|
|
if ( !!outArr ) {
|
|
XP_ASSERT( outArr->format == FORMAT_LOC );
|
|
SMSMsgLoc* msg = &outArr->u.msgsLoc[0];
|
|
XP_ASSERT( msg->gameID == gameID );
|
|
XP_ASSERT( msg->cmd == DATA );
|
|
// XP_LOGF( "%s(): got msgID %d", __func__, msg->msgID );
|
|
XP_ASSERT( outArr->nMsgs == 1 );
|
|
XP_ASSERT( 0 == memcmp(buf, msg->data, (ii + 1) * 30) );
|
|
smsproto_freeMsgArray( state, outArr );
|
|
|
|
smsproto_freeMsgArray( state, arrs[ii] );
|
|
arrs[ii] = NULL;
|
|
}
|
|
}
|
|
}
|
|
if (!haveOne) {
|
|
break;
|
|
}
|
|
}
|
|
|
|
/* Now let's send a bunch of small messages that should get combined */
|
|
for ( int nUsed = 0; ; ++nUsed ) {
|
|
XP_U16 waitSecs;
|
|
SMSMsgArray* sendArr = smsproto_prepOutbound( state, DATA, gameID, &buf[nUsed],
|
|
smallSiz, phones[0], port,
|
|
XP_FALSE, &waitSecs );
|
|
if ( sendArr == NULL ) {
|
|
XP_LOGF( "%s(): msg[%d] of len %d sent; still not ready", __func__, nUsed, smallSiz );
|
|
continue;
|
|
}
|
|
|
|
XP_ASSERT( waitSecs == 0 );
|
|
XP_ASSERT( sendArr->format == FORMAT_NET );
|
|
int totalBack = 0;
|
|
for ( int jj = 0; jj < sendArr->nMsgs; ++jj ) {
|
|
SMSMsgArray* recvArr = smsproto_prepInbound( state, phones[0], port,
|
|
sendArr->u.msgsNet[jj].data,
|
|
sendArr->u.msgsNet[jj].len );
|
|
|
|
if ( !!recvArr ) {
|
|
XP_ASSERT( recvArr->format == FORMAT_LOC );
|
|
XP_LOGF( "%s(): got %d msgs (from %d)", __func__, recvArr->nMsgs, nUsed + 1 );
|
|
for ( int kk = 0; kk < recvArr->nMsgs; ++kk ) {
|
|
SMSMsgLoc* msg = &recvArr->u.msgsLoc[kk];
|
|
// XP_LOGF( "%s(): got msgID %d", __func__, msg->msgID );
|
|
XP_ASSERT( msg->gameID == gameID );
|
|
XP_ASSERT( msg->cmd == DATA );
|
|
XP_ASSERT( msg->len == smallSiz );
|
|
XP_ASSERT( 0 == memcmp( msg->data, &buf[totalBack], smallSiz ) );
|
|
++totalBack;
|
|
}
|
|
|
|
smsproto_freeMsgArray( state, recvArr );
|
|
}
|
|
}
|
|
XP_ASSERT( forceOld || totalBack == nUsed + 1 );
|
|
XP_LOGF( "%s(): %d messages checked out", __func__, totalBack );
|
|
smsproto_freeMsgArray( state, sendArr );
|
|
break;
|
|
}
|
|
|
|
/* Now let's add a too-long message and unpack only the first part. Make
|
|
sure it's cleaned up correctly */
|
|
XP_U16 waitSecs;
|
|
SMSMsgArray* arr = smsproto_prepOutbound( state, DATA, gameID, buf, 200, "33333",
|
|
port, XP_TRUE, &waitSecs );
|
|
XP_ASSERT( !!arr && arr->nMsgs > 1 );
|
|
/* add only part 1 */
|
|
SMSMsgArray* out = smsproto_prepInbound( state, "33333", port, arr->u.msgsNet[0].data,
|
|
arr->u.msgsNet[0].len );
|
|
XP_ASSERT( !out );
|
|
smsproto_freeMsgArray( state, arr );
|
|
|
|
/* Try the no-buffer messages */
|
|
XP_LOGF( "%s(): trying DEATH", __func__ );
|
|
arr = smsproto_prepOutbound( state, DEATH, gameID, NULL, 0, "33333",
|
|
port, XP_TRUE, &waitSecs );
|
|
XP_ASSERT( arr->format == FORMAT_NET );
|
|
out = smsproto_prepInbound( state, "33333", port,
|
|
arr->u.msgsNet[0].data,
|
|
arr->u.msgsNet[0].len );
|
|
XP_ASSERT( out->format == FORMAT_LOC );
|
|
XP_ASSERT( out->u.msgsLoc[0].cmd == DEATH );
|
|
XP_ASSERT( out->u.msgsLoc[0].gameID == gameID );
|
|
smsproto_freeMsgArray( state, arr );
|
|
smsproto_freeMsgArray( state, out );
|
|
XP_LOGF( "%s(): DEATH checked out", __func__ );
|
|
|
|
/* Test port mismatch */
|
|
arr = smsproto_prepOutbound( state, DEATH, gameID, NULL, 0, "33333",
|
|
port, XP_TRUE, &waitSecs );
|
|
XP_ASSERT( arr->format == FORMAT_NET );
|
|
out = smsproto_prepInbound( state, "33333", port + 1,
|
|
arr->u.msgsNet[0].data,
|
|
arr->u.msgsNet[0].len );
|
|
XP_ASSERT( out == NULL );
|
|
smsproto_freeMsgArray( state, arr );
|
|
XP_LOGF( "%s(): mismatched port test done", __func__ );
|
|
|
|
/* now a message that's unpacked across multiple sessions to test store/load */
|
|
XP_LOGF( "%s(): testing store/restore", __func__ );
|
|
arr = smsproto_prepOutbound( state, DATA, gameID, (XP_U8*)buf, 200, "33333",
|
|
port, XP_TRUE, &waitSecs );
|
|
for ( int ii = 0; ii < arr->nMsgs; ++ii ) {
|
|
SMSMsgArray* out = smsproto_prepInbound( state, "33333", port,
|
|
arr->u.msgsNet[ii].data,
|
|
arr->u.msgsNet[ii].len );
|
|
if ( !!out ) {
|
|
XP_ASSERT( out->nMsgs == 1);
|
|
XP_ASSERT( out->format == FORMAT_LOC );
|
|
XP_LOGF( "%s(): got the message on the %dth loop", __func__, ii );
|
|
XP_ASSERT( out->u.msgsLoc[0].len == 200 );
|
|
XP_ASSERT( 0 == memcmp( out->u.msgsLoc[0].data, buf, 200 ) );
|
|
smsproto_freeMsgArray( state, out );
|
|
break;
|
|
}
|
|
smsproto_free( state ); /* give it a chance to store state */
|
|
state = smsproto_init( mpool, dutil );
|
|
}
|
|
|
|
/* Really bad to pass a different state than was created with, but now
|
|
since only mpool is used and it's the same for all states, let it
|
|
go. */
|
|
smsproto_freeMsgArray( state, arr ); /* give it a chance to store state */
|
|
|
|
smsproto_free( state );
|
|
LOG_RETURN_VOID();
|
|
}
|
|
#endif
|