mirror of
git://xwords.git.sourceforge.net/gitroot/xwords/xwords
synced 2025-01-03 23:04:08 +01:00
1181e908dc
When rematching, some users have a convention that e.g. lowest scoring player in the "parent" game goes first. So allow that, providing the choice on each rematch until a default has been chosen. Support changing that default in a new prefs setting. The place I chose to enforce the order was on the host as invitees are registering and being assigned slots. But by then there's no longer any connection to the game that was rematched, e.g. to use its scores. So during the rematched game creation process I create and store with the new game the necessary ordering information. For the 3-and-4 device case, it was also necessary to tweak the information about other guests that the host sends guests (added during earlier work on rematching.)
755 lines
20 KiB
C
755 lines
20 KiB
C
/* -*-mode: C; compile-command: "cd ../linux && make MEMDEBUG=TRUE"; -*- */
|
|
/*
|
|
* Copyright 2001-2009 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 "strutils.h"
|
|
#include "xwstream.h"
|
|
#include "mempool.h"
|
|
#include "xptypes.h"
|
|
|
|
#ifdef CPLUS
|
|
extern "C" {
|
|
#endif
|
|
|
|
XP_U16
|
|
bitsForMax( XP_U32 nn )
|
|
{
|
|
XP_U16 result = 0;
|
|
XP_ASSERT( nn > 0 );
|
|
|
|
while ( nn != 0 ) {
|
|
nn >>= 1;
|
|
++result;
|
|
}
|
|
|
|
return result;
|
|
} /* bitsForMax */
|
|
|
|
static void
|
|
tilesToStream( XWStreamCtxt* stream, const Tile* tiles, XP_U16 nTiles )
|
|
{
|
|
while ( nTiles-- ) {
|
|
stream_putBits( stream, TILE_NBITS, *tiles++ );
|
|
}
|
|
} /* tilesToStream */
|
|
|
|
void
|
|
traySetToStream( XWStreamCtxt* stream, const TrayTileSet* ts )
|
|
{
|
|
XP_U16 nTiles = ts->nTiles;
|
|
stream_putBits( stream, tilesNBits(stream), nTiles );
|
|
tilesToStream( stream, ts->tiles, nTiles );
|
|
} /* traySetFromStream */
|
|
|
|
static void
|
|
tilesFromStream( XWStreamCtxt* stream, Tile* tiles, XP_U16 nTiles )
|
|
{
|
|
while ( nTiles-- ) {
|
|
*tiles++ = (Tile)stream_getBits( stream, TILE_NBITS );
|
|
}
|
|
} /* tilesFromStream */
|
|
|
|
void
|
|
scoresToStream( XWStreamCtxt* stream, XP_U16 nScores, const XP_U16* scores )
|
|
{
|
|
if ( 0 < nScores ) {
|
|
XP_U16 maxScore = 1; /* 0 will confuse bitsForMax */
|
|
for ( XP_U16 ii = 0; ii < nScores; ++ii ) {
|
|
XP_U16 score = scores[ii];
|
|
if ( score > maxScore ) {
|
|
maxScore = score;
|
|
}
|
|
}
|
|
|
|
XP_U16 bits = bitsForMax( maxScore );
|
|
stream_putBits( stream, 4, bits );
|
|
for ( XP_U16 ii = 0; ii < nScores; ++ii ) {
|
|
stream_putBits( stream, bits, scores[ii] );
|
|
}
|
|
}
|
|
}
|
|
|
|
void
|
|
scoresFromStream( XWStreamCtxt* stream, XP_U16 nScores, XP_U16* scores )
|
|
{
|
|
if ( 0 < nScores ) {
|
|
XP_U16 bits = (XP_U16)stream_getBits( stream, 4 );
|
|
for ( XP_U16 ii = 0; ii < nScores; ++ii ) {
|
|
scores[ii] = stream_getBits( stream, bits );
|
|
}
|
|
}
|
|
}
|
|
|
|
void
|
|
traySetFromStream( XWStreamCtxt* stream, TrayTileSet* ts )
|
|
{
|
|
XP_U16 nTiles = (XP_U16)stream_getBits( stream, tilesNBits( stream ) );
|
|
tilesFromStream( stream, ts->tiles, nTiles );
|
|
ts->nTiles = (XP_U8)nTiles;
|
|
} /* traySetFromStream */
|
|
|
|
#ifdef DEBUG
|
|
void
|
|
assertSorted( const MoveInfo* mi )
|
|
{
|
|
for ( XP_U16 ii = 1; ii < mi->nTiles; ++ii ) {
|
|
XP_ASSERT( mi->tiles[ii-1].varCoord < mi->tiles[ii].varCoord );
|
|
}
|
|
}
|
|
#endif
|
|
|
|
void
|
|
moveInfoToStream( XWStreamCtxt* stream, const MoveInfo* mi, XP_U16 bitsPerTile )
|
|
{
|
|
#ifdef DEBUG
|
|
/* XP_UCHAR buf[64] = {0}; */
|
|
/* XP_U16 offset = 0; */
|
|
#endif
|
|
assertSorted( mi );
|
|
|
|
stream_putBits( stream, tilesNBits( stream ), mi->nTiles );
|
|
stream_putBits( stream, NUMCOLS_NBITS_5, mi->commonCoord );
|
|
stream_putBits( stream, 1, mi->isHorizontal );
|
|
|
|
XP_ASSERT( bitsPerTile == 5 || bitsPerTile == 6 );
|
|
for ( XP_U16 ii = 0; ii < mi->nTiles; ++ii ) {
|
|
stream_putBits( stream, NUMCOLS_NBITS_5, mi->tiles[ii].varCoord );
|
|
|
|
Tile tile = mi->tiles[ii].tile;
|
|
#ifdef DEBUG
|
|
/* offset += XP_SNPRINTF( &buf[offset], VSIZE(buf)-offset, "%x,", tile ); */
|
|
#endif
|
|
stream_putBits( stream, bitsPerTile, tile & TILE_VALUE_MASK );
|
|
stream_putBits( stream, 1, (tile & TILE_BLANK_BIT) != 0 );
|
|
}
|
|
// XP_LOGF( "%s(): tiles: %s", __func__, buf );
|
|
}
|
|
|
|
void
|
|
moveInfoFromStream( XWStreamCtxt* stream, MoveInfo* mi, XP_U16 bitsPerTile )
|
|
{
|
|
#ifdef DEBUG
|
|
/* XP_UCHAR buf[64] = {0}; */
|
|
/* XP_U16 offset = 0; */
|
|
#endif
|
|
mi->nTiles = stream_getBits( stream, tilesNBits( stream ) );
|
|
XP_ASSERT( mi->nTiles <= MAX_TRAY_TILES );
|
|
mi->commonCoord = stream_getBits( stream, NUMCOLS_NBITS_5 );
|
|
mi->isHorizontal = stream_getBits( stream, 1 );
|
|
for ( XP_U16 ii = 0; ii < mi->nTiles; ++ii ) {
|
|
mi->tiles[ii].varCoord = stream_getBits( stream, NUMCOLS_NBITS_5 );
|
|
Tile tile = stream_getBits( stream, bitsPerTile );
|
|
if ( 0 != stream_getBits( stream, 1 ) ) {
|
|
tile |= TILE_BLANK_BIT;
|
|
}
|
|
mi->tiles[ii].tile = tile;
|
|
#ifdef DEBUG
|
|
/* offset += XP_SNPRINTF( &buf[offset], VSIZE(buf)-offset, "%x,", tile ); */
|
|
#endif
|
|
}
|
|
assertSorted( mi );
|
|
// XP_LOGF( "%s(): tiles: %s", __func__, buf );
|
|
}
|
|
|
|
void
|
|
removeTile( TrayTileSet* tiles, XP_U16 index )
|
|
{
|
|
XP_U16 ii;
|
|
--tiles->nTiles;
|
|
for ( ii = index; ii < tiles->nTiles; ++ii ) {
|
|
tiles->tiles[ii] = tiles->tiles[ii+1];
|
|
}
|
|
}
|
|
|
|
void
|
|
sortTiles( TrayTileSet* dest, const TrayTileSet* src, XP_U16 skip )
|
|
{
|
|
XP_ASSERT( src->nTiles >= skip );
|
|
TrayTileSet tmp = *src;
|
|
|
|
/* Copy in the ones we're not sorting */
|
|
dest->nTiles = skip;
|
|
if ( 0 < skip ) {
|
|
XP_MEMCPY( &dest->tiles, &tmp.tiles, skip * sizeof(tmp.tiles[0]) );
|
|
}
|
|
|
|
while ( skip < tmp.nTiles ) {
|
|
XP_U16 ii, smallest;
|
|
for ( smallest = ii = skip; ii < tmp.nTiles; ++ii ) {
|
|
if ( tmp.tiles[ii] < tmp.tiles[smallest] ) {
|
|
smallest = ii;
|
|
}
|
|
}
|
|
dest->tiles[dest->nTiles++] = tmp.tiles[smallest];
|
|
removeTile( &tmp, smallest );
|
|
}
|
|
}
|
|
|
|
#if 0
|
|
static void
|
|
signedToStream( XWStreamCtxt* stream, XP_U16 nBits, XP_S32 num )
|
|
{
|
|
XP_Bool negative = num < 0;
|
|
stream_putBits( stream, 1, negative );
|
|
if ( negative ) {
|
|
num *= -1;
|
|
}
|
|
stream_putBits( stream, nBits, num );
|
|
} /* signedToStream */
|
|
|
|
XP_S32
|
|
signedFromStream( XWStreamCtxt* stream, XP_U16 nBits )
|
|
{
|
|
XP_S32 result;
|
|
XP_Bool negative = stream_getBits( stream, 1 );
|
|
result = stream_getBits( stream, nBits );
|
|
if ( negative ) {
|
|
result *= -1;
|
|
}
|
|
return result;
|
|
} /* signedFromStream */
|
|
#endif
|
|
|
|
XP_UCHAR*
|
|
p_stringFromStream( MPFORMAL XWStreamCtxt* stream
|
|
#ifdef MEM_DEBUG
|
|
, const char* file, const char* func, XP_U32 lineNo
|
|
#endif
|
|
)
|
|
{
|
|
XP_U16 version = stream_getVersion( stream );
|
|
XP_U32 len = version < STREAM_VERS_NORELAY ? stream_getU8( stream )
|
|
: stream_getU32VL( stream );
|
|
|
|
XP_UCHAR* str = NULL;
|
|
if ( 0 < len ) {
|
|
#ifdef MEM_DEBUG
|
|
str = mpool_alloc( mpool, len + 1, file, func, lineNo );
|
|
#else
|
|
str = (XP_UCHAR*)XP_MALLOC( mpool, len + 1 );
|
|
#endif
|
|
stream_getBytes( stream, str, len );
|
|
str[len] = '\0';
|
|
}
|
|
return str;
|
|
} /* makeStringFromStream */
|
|
|
|
XP_U16
|
|
stringFromStreamHereImpl( XWStreamCtxt* stream, XP_UCHAR* buf, XP_U16 buflen
|
|
#ifdef DEBUG
|
|
, const char* func, int line
|
|
#endif
|
|
)
|
|
{
|
|
XP_U16 version = stream_getVersion( stream );
|
|
|
|
XP_U32 len = version < STREAM_VERS_NORELAY ? stream_getU8( stream )
|
|
: stream_getU32VL( stream );
|
|
if ( len > 0 ) {
|
|
if ( buflen <= len ) {
|
|
XP_LOGFF( "BAD: buflen %d < len %d (from %s(), line %d)", buflen, len, func, line );
|
|
XP_ASSERT(0);
|
|
}
|
|
if ( len >= buflen ) {
|
|
/* better to leave stream in bad state than overwrite stack */
|
|
len = buflen - 1;
|
|
}
|
|
stream_getBytes( stream, buf, len );
|
|
}
|
|
buf[len] = '\0';
|
|
return len;
|
|
}
|
|
|
|
void
|
|
stringToStream( XWStreamCtxt* stream, const XP_UCHAR* str )
|
|
{
|
|
XP_U16 version = stream_getVersion( stream );
|
|
|
|
XP_U32 len = str == NULL? 0: XP_STRLEN( str );
|
|
if ( version < STREAM_VERS_NORELAY ) {
|
|
if ( len > 0xFF ) {
|
|
XP_LOGFF( "truncating string '%s', dropping len from %d to %d",
|
|
str, len, 0xFF );
|
|
XP_ASSERT(0);
|
|
len = 0xFF;
|
|
}
|
|
stream_putU8( stream, (XP_U8)len );
|
|
} else {
|
|
stream_putU32VL( stream, len );
|
|
}
|
|
stream_putBytes( stream, str, len );
|
|
} /* putStringToStream */
|
|
|
|
XP_Bool
|
|
stream_gotU8( XWStreamCtxt* stream, XP_U8* ptr )
|
|
{
|
|
XP_Bool success = sizeof(*ptr) <= stream_getSize( stream );
|
|
if ( success ) {
|
|
*ptr = stream_getU8( stream );
|
|
}
|
|
return success;
|
|
}
|
|
|
|
XP_Bool
|
|
stream_gotU16( XWStreamCtxt* stream, XP_U16* ptr )
|
|
{
|
|
XP_Bool success = sizeof(*ptr) <= stream_getSize( stream );
|
|
if ( success ) {
|
|
*ptr = stream_getU16( stream );
|
|
}
|
|
return success;
|
|
}
|
|
|
|
XP_Bool
|
|
stream_gotU32( XWStreamCtxt* stream, XP_U32* ptr )
|
|
{
|
|
XP_Bool success = sizeof(*ptr) <= stream_getSize( stream );
|
|
if ( success ) {
|
|
*ptr = stream_getU32( stream );
|
|
}
|
|
return success;
|
|
}
|
|
|
|
XP_Bool
|
|
stream_gotBytes( XWStreamCtxt* stream, void* ptr, XP_U16 len )
|
|
{
|
|
XP_Bool success = len <= stream_getSize( stream );
|
|
if ( success ) {
|
|
stream_getBytes( stream, ptr, len );
|
|
}
|
|
return success;
|
|
}
|
|
|
|
/*****************************************************************************
|
|
*
|
|
****************************************************************************/
|
|
XP_UCHAR*
|
|
p_copyString( MPFORMAL const XP_UCHAR* instr
|
|
#ifdef MEM_DEBUG
|
|
, const char* file, const char* func, XP_U32 lineNo
|
|
#endif
|
|
)
|
|
{
|
|
XP_UCHAR* result = (XP_UCHAR*)NULL;
|
|
if ( !!instr ) {
|
|
XP_U16 len = 1 + XP_STRLEN( (const char*)instr );
|
|
#ifdef MEM_DEBUG
|
|
result = mpool_alloc( mpool, len, file, func, lineNo );
|
|
#else
|
|
result = XP_MALLOC( ignore, len );
|
|
#endif
|
|
|
|
XP_ASSERT( !!result );
|
|
XP_MEMCPY( result, instr, len );
|
|
}
|
|
return result;
|
|
} /* copyString */
|
|
|
|
void
|
|
p_replaceStringIfDifferent( MPFORMAL XP_UCHAR** curLoc, const XP_UCHAR* newStr
|
|
#ifdef MEM_DEBUG
|
|
, const char* file, const char* func, XP_U32 lineNo
|
|
#endif
|
|
)
|
|
{
|
|
XP_UCHAR* curStr = *curLoc;
|
|
|
|
if ( !!newStr && !!curStr &&
|
|
(0 == XP_STRCMP( (const char*)curStr, (const char*)newStr ) ) ) {
|
|
/* do nothing; we're golden */
|
|
} else {
|
|
XP_FREEP( mpool, &curStr );
|
|
#ifdef MEM_DEBUG
|
|
curStr = p_copyString( mpool, newStr, file, func, lineNo );
|
|
#else
|
|
curStr = p_copyString( newStr );
|
|
#endif
|
|
}
|
|
|
|
*curLoc = curStr;
|
|
} /* replaceStringIfDifferent */
|
|
|
|
void
|
|
insetRect( XP_Rect* rect, XP_S16 byWidth, XP_S16 byHeight )
|
|
{
|
|
rect->width -= byWidth * 2;
|
|
rect->left += byWidth;
|
|
rect->height -= byHeight * 2;
|
|
rect->top += byHeight;
|
|
}
|
|
|
|
XP_U32
|
|
augmentHash( XP_U32 hash, const XP_U8* ptr, XP_U16 len )
|
|
{
|
|
// see http://en.wikipedia.org/wiki/Jenkins_hash_function
|
|
for ( XP_U16 ii = 0; ii < len; ++ii ) {
|
|
hash += *ptr++;
|
|
hash += (hash << 10);
|
|
hash ^= (hash >> 6);
|
|
}
|
|
return hash;
|
|
}
|
|
|
|
XP_U32
|
|
finishHash( XP_U32 hash )
|
|
{
|
|
hash += (hash << 3);
|
|
hash ^= (hash >> 11);
|
|
hash += (hash << 15);
|
|
return hash;
|
|
}
|
|
|
|
XP_U16
|
|
tilesNBits( const XWStreamCtxt* stream )
|
|
{
|
|
XP_U16 version = stream_getVersion( stream );
|
|
XP_ASSERT( 0 < version );
|
|
if ( 0 == version ) {
|
|
XP_LOGFF( "version is 0" );
|
|
}
|
|
XP_U16 result = STREAM_VERS_NINETILES <= version
|
|
? NTILES_NBITS_9 : NTILES_NBITS_7;
|
|
return result;
|
|
}
|
|
|
|
/* sMap: Exists as a backup until everybody's running code that knows isoCode
|
|
is in wordlists. */
|
|
static struct {
|
|
XP_LangCode lc;
|
|
XP_UCHAR* isoCode;
|
|
} sMap[] = {
|
|
{ .lc = 0x01, .isoCode = "en", },
|
|
{ .lc = 0x02, .isoCode = "fr", },
|
|
{ .lc = 0x03, .isoCode = "de", },
|
|
{ .lc = 0x04, .isoCode = "tr", },
|
|
{ .lc = 0x05, .isoCode = "ar", },
|
|
{ .lc = 0x06, .isoCode = "es", },
|
|
{ .lc = 0x07, .isoCode = "sv", },
|
|
{ .lc = 0x08, .isoCode = "pl", },
|
|
{ .lc = 0x09, .isoCode = "da", },
|
|
{ .lc = 0x0A, .isoCode = "it", },
|
|
{ .lc = 0x0B, .isoCode = "nl", },
|
|
{ .lc = 0x0C, .isoCode = "ca", },
|
|
{ .lc = 0x0D, .isoCode = "pt", },
|
|
{ .lc = 0x0F, .isoCode = "ru", },
|
|
{ .lc = 0x11, .isoCode = "cs", },
|
|
{ .lc = 0x12, .isoCode = "el", },
|
|
{ .lc = 0x13, .isoCode = "sk", },
|
|
{ .lc = 0x14, .isoCode = "hu", },
|
|
{ .lc = 0x15, .isoCode = "ro", },
|
|
{ .lc = 0x19, .isoCode = "fi", },
|
|
};
|
|
|
|
XP_Bool
|
|
haveLocaleToLc( const XP_UCHAR* isoCode, XP_LangCode* lc )
|
|
{
|
|
XP_Bool result = XP_FALSE;
|
|
for ( int ii = 0; !result && ii < VSIZE(sMap); ++ii ) {
|
|
if ( 0 == XP_STRCMP( isoCode, sMap[ii].isoCode ) ) {
|
|
result = XP_TRUE;
|
|
*lc = sMap[ii].lc;
|
|
}
|
|
}
|
|
return result;
|
|
}
|
|
|
|
const XP_UCHAR*
|
|
lcToLocale( XP_LangCode lc )
|
|
{
|
|
const XP_UCHAR* result = NULL;
|
|
for ( int ii = 0; NULL == result && ii < VSIZE(sMap); ++ii ) {
|
|
if ( lc == sMap[ii].lc ) {
|
|
result = sMap[ii].isoCode;
|
|
}
|
|
}
|
|
if ( !result ) {
|
|
XP_LOGFF( "(%d/0x%x) => NULL", lc, lc );
|
|
}
|
|
return result;
|
|
}
|
|
|
|
/*
|
|
* A wrapper for printing etc. potentially null strings.
|
|
*/
|
|
XP_UCHAR*
|
|
emptyStringIfNull( XP_UCHAR* str )
|
|
{
|
|
return !!str? str : (XP_UCHAR*)"";
|
|
} /* emptyStringIfNull */
|
|
|
|
XP_Bool
|
|
randIntArray( XP_U16* rnums, XP_U16 count )
|
|
{
|
|
XP_Bool changed = XP_FALSE;
|
|
XP_U16 ii;
|
|
|
|
for ( ii = 0; ii < count; ++ii ) {
|
|
rnums[ii] = ii;
|
|
}
|
|
|
|
for ( ii = count; ii > 0 ; ) {
|
|
XP_U16 rIndex = ((XP_U16)XP_RANDOM()) % ii;
|
|
if ( --ii != rIndex ) {
|
|
XP_U16 tmp = rnums[rIndex];
|
|
rnums[rIndex] = rnums[ii];
|
|
rnums[ii] = tmp;
|
|
if ( !changed ) {
|
|
changed = XP_TRUE;
|
|
}
|
|
}
|
|
}
|
|
|
|
return changed;
|
|
} /* randIntArray */
|
|
|
|
XP_U16
|
|
countBits( XP_U32 mask )
|
|
{
|
|
XP_U16 result = 0;
|
|
while ( 0 != mask ) {
|
|
++result;
|
|
mask &= mask - 1;
|
|
}
|
|
return result;
|
|
}
|
|
|
|
#ifdef XWFEATURE_BASE64
|
|
/* base-64 encode binary data as a message legal for SMS. See
|
|
* http://www.ietf.org/rfc/rfc2045.txt for the algorithm. glib uses this and
|
|
* so it's not needed on linux, but unless all platforms provided identical
|
|
* implementations it's needed for messages to be cross-platform.
|
|
*/
|
|
|
|
static const XP_UCHAR*
|
|
getSMSTable( void )
|
|
{
|
|
return "ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/";
|
|
}
|
|
#define PADCHAR '='
|
|
|
|
static void
|
|
bitsToChars( const XP_U8* bytesIn, XP_U16 nValidBytes, XP_UCHAR* out,
|
|
XP_U16* outlen )
|
|
{
|
|
XP_U16 nValidSextets = ((nValidBytes * 8) + 5) / 6; /* +5: round up */
|
|
XP_U8 local[4] = { 0, 0, 0, 0 };
|
|
XP_U16 bits[4];
|
|
XP_MEMCPY( local, bytesIn, nValidBytes );
|
|
/* top 6 bits of first byte */
|
|
bits[0] = local[0] >> 2;
|
|
/* bottom 2 bits of first byte, top four of second */
|
|
bits[1] = ((local[0] << 4) & 0x30 ) | (local[1] >> 4);
|
|
/* bottom four bits of second byte, top two of third */
|
|
bits[2] = ((local[1] << 2) & 0x3C) | (local[2] >> 6);
|
|
/* bottom six bits of third */
|
|
bits[3] = local[2] & 0x3F;
|
|
|
|
const XP_UCHAR* table = getSMSTable();
|
|
|
|
XP_U16 ii;
|
|
for ( ii = 0; ii < 4; ++ii ) {
|
|
XP_UCHAR ch;
|
|
if ( ii < nValidSextets ) {
|
|
XP_U16 index = bits[ii];
|
|
ch = table[index];
|
|
} else {
|
|
ch = PADCHAR;
|
|
}
|
|
out[(*outlen)++] = ch;
|
|
}
|
|
} /* bitsToChars */
|
|
|
|
void
|
|
binToSms( XP_UCHAR* out, XP_U16* outlenp, const XP_U8* in, const XP_U16 inlen )
|
|
{
|
|
XP_U16 inConsumed;
|
|
XP_U16 outlen = 0;
|
|
|
|
for ( inConsumed = 0; ; /*inConsumed += 3*/ ) {
|
|
XP_U16 validBytes = XP_MIN( 3, inlen - inConsumed );
|
|
bitsToChars( &in[inConsumed], validBytes, out, &outlen );
|
|
XP_ASSERT( outlen <= *outlenp );
|
|
|
|
inConsumed += 3;
|
|
if ( inConsumed >= inlen ) {
|
|
break;
|
|
}
|
|
}
|
|
XP_ASSERT( outlen < *outlenp );
|
|
out[outlen] = '\0';
|
|
*outlenp = outlen;
|
|
XP_ASSERT( *outlenp >= inlen );
|
|
} /* binToSms */
|
|
|
|
/* Return false if illegal, e.g. contains bad characters.
|
|
*/
|
|
|
|
static XP_U8
|
|
findRank( XP_UCHAR ch )
|
|
{
|
|
XP_U8 rank;
|
|
if ( ch == PADCHAR ) {
|
|
rank = 0;
|
|
} else {
|
|
const XP_UCHAR* table = getSMSTable();
|
|
for ( rank = 0; rank < 64; ++rank ) {
|
|
if ( table[rank] == ch ) {
|
|
break;
|
|
}
|
|
}
|
|
XP_ASSERT( rank < 64 );
|
|
}
|
|
return rank;
|
|
}
|
|
|
|
/* This function stolen from glib file glib/gbase64.c. It's also GPL'd, so
|
|
* that may not matter. But does my copyright need to change? PENDING
|
|
*
|
|
* Also, need to check there's space before writing! PENDING
|
|
*/
|
|
XP_Bool
|
|
smsToBin( XP_U8* out, XP_U16* outlenp, const XP_UCHAR* sms, XP_U16 smslen )
|
|
{
|
|
const XP_UCHAR* inptr;
|
|
XP_U8* outptr = out;
|
|
const XP_UCHAR* smsend = sms + smslen;
|
|
XP_U8 ch, rank;
|
|
XP_U8 last[2];
|
|
unsigned int vv = 0;
|
|
int ii = 0;
|
|
|
|
inptr = sms;
|
|
last[0] = last[1] = 0;
|
|
while ( inptr < smsend ) {
|
|
ch = *inptr++;
|
|
rank = findRank( ch );
|
|
|
|
last[1] = last[0];
|
|
last[0] = ch;
|
|
vv = (vv<<6) | rank;
|
|
if ( ++ii == 4 ) {
|
|
*outptr++ = vv >> 16;
|
|
if (last[1] != PADCHAR ) {
|
|
*outptr++ = vv >> 8;
|
|
}
|
|
if (last[0] != PADCHAR ) {
|
|
*outptr++ = vv;
|
|
}
|
|
ii = 0;
|
|
}
|
|
}
|
|
|
|
XP_ASSERT( *outlenp >= (outptr - out) );
|
|
*outlenp = outptr - out;
|
|
|
|
return XP_TRUE;
|
|
} /* smsToBin */
|
|
|
|
#endif
|
|
|
|
XP_UCHAR*
|
|
formatMQTTDevID( const MQTTDevID* devid, XP_UCHAR* buf, XP_U16 bufLen )
|
|
{
|
|
XP_SNPRINTF( buf, bufLen, MQTTDevID_FMT, *devid );
|
|
return buf;
|
|
}
|
|
|
|
XP_UCHAR*
|
|
formatMQTTDevTopic( const MQTTDevID* devid, XP_UCHAR* buf, XP_U16 bufLen )
|
|
{
|
|
XP_SNPRINTF( buf, bufLen, MQTTTopic_FMT, *devid );
|
|
// LOG_RETURNF( "%s", buf );
|
|
return buf;
|
|
}
|
|
|
|
XP_UCHAR*
|
|
formatMQTTCtrlTopic( const MQTTDevID* devid, XP_UCHAR* buf, XP_U16 bufLen )
|
|
{
|
|
XP_SNPRINTF( buf, bufLen, MQTTCtrlTopic_FMT, *devid );
|
|
return buf;
|
|
}
|
|
|
|
XP_Bool
|
|
strToMQTTCDevID( const XP_UCHAR* str, MQTTDevID* result )
|
|
{
|
|
XP_Bool success = XP_FALSE;
|
|
if ( 16 == strlen(str) ) {
|
|
MQTTDevID tmp;
|
|
int nMatched = sscanf( str, MQTTDevID_FMT, &tmp );
|
|
success = nMatched == 1;
|
|
if ( success ) {
|
|
*result = tmp;
|
|
}
|
|
}
|
|
return success;
|
|
}
|
|
|
|
#ifdef DEBUG
|
|
#define NUM_PER_LINE 8
|
|
void
|
|
log_hex( const XP_U8* memp, XP_U16 len, const char* tag )
|
|
{
|
|
XP_LOGF( "%s(len=%d[0x%x])", __func__, len, len );
|
|
const char* hex = "0123456789ABCDEF";
|
|
XP_U16 ii, jj;
|
|
XP_U16 offset = 0;
|
|
|
|
while ( offset < len ) {
|
|
XP_UCHAR buf[128];
|
|
XP_UCHAR vals[NUM_PER_LINE*3];
|
|
XP_UCHAR* valsp = vals;
|
|
XP_UCHAR chars[NUM_PER_LINE+1];
|
|
XP_UCHAR* charsp = chars;
|
|
XP_U16 oldOffset = offset;
|
|
|
|
for ( ii = 0; ii < NUM_PER_LINE && offset < len; ++ii ) {
|
|
XP_U8 byte = memp[offset];
|
|
for ( jj = 0; jj < 2; ++jj ) {
|
|
*valsp++ = hex[(byte & 0xF0) >> 4];
|
|
byte <<= 4;
|
|
}
|
|
*valsp++ = ':';
|
|
|
|
byte = memp[offset];
|
|
if ( (byte >= 'A' && byte <= 'Z')
|
|
|| (byte >= 'a' && byte <= 'z')
|
|
|| (byte >= '0' && byte <= '9') ) {
|
|
/* keep it */
|
|
} else {
|
|
byte = '.';
|
|
}
|
|
*charsp++ = byte;
|
|
++offset;
|
|
}
|
|
*(valsp-1) = '\0'; /* -1 to overwrite ':' */
|
|
*charsp = '\0';
|
|
|
|
if ( (NULL == tag) || (XP_STRLEN(tag) + sizeof(vals) >= sizeof(buf)) ) {
|
|
tag = "<tag>";
|
|
}
|
|
XP_SNPRINTF( buf, sizeof(buf), "%s[%.3d]: %-24s %s", tag, oldOffset,
|
|
vals, chars );
|
|
XP_LOGF( "%s", buf );
|
|
}
|
|
}
|
|
#endif
|
|
|
|
#ifdef CPLUS
|
|
}
|
|
#endif
|