mirror of
git://xwords.git.sourceforge.net/gitroot/xwords/xwords
synced 2025-01-14 08:01:38 +01:00
1067 lines
35 KiB
C
1067 lines
35 KiB
C
/* -*- compile-command: "cd ../linux && make -j3 MEMDEBUG=TRUE"; -*- */
|
|
/*
|
|
* Copyright 2001-2011 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 "game.h"
|
|
#include "dictnry.h"
|
|
#include "strutils.h"
|
|
#include "nli.h"
|
|
#include "dbgutil.h"
|
|
|
|
#ifdef CPLUS
|
|
extern "C" {
|
|
#endif
|
|
|
|
#define FLAG_HASCOMMS 0x01
|
|
|
|
#ifdef DEBUG
|
|
static void
|
|
assertUtilOK( XW_UtilCtxt* util )
|
|
{
|
|
UtilVtable* vtable;
|
|
XP_U16 nSlots;
|
|
XP_ASSERT( !!util );
|
|
vtable = util->vtable;
|
|
nSlots = sizeof(vtable) / 4;
|
|
while ( nSlots-- ) {
|
|
void* fptr = ((void**)vtable)[nSlots];
|
|
XP_ASSERT( !!fptr );
|
|
}
|
|
} /* assertUtilOK */
|
|
#else
|
|
# define assertUtilOK(u)
|
|
#endif
|
|
|
|
#ifdef XWFEATURE_CHANGEDICT
|
|
static void gi_setDict( MPFORMAL CurGameInfo* gi, const DictionaryCtxt* dict );
|
|
#endif
|
|
|
|
static void
|
|
checkServerRole( CurGameInfo* gi, XP_U16* nPlayersHere,
|
|
XP_U16* nPlayersTotal )
|
|
{
|
|
if ( !!gi ) {
|
|
XP_U16 ii, remoteCount = 0;
|
|
|
|
if ( SERVER_STANDALONE != gi->serverRole ) {
|
|
for ( ii = 0; ii < gi->nPlayers; ++ii ) {
|
|
if ( !gi->players[ii].isLocal ) {
|
|
++remoteCount;
|
|
}
|
|
}
|
|
|
|
/* I think this error is caught in nwgamest.c now */
|
|
XP_ASSERT( remoteCount > 0 );
|
|
if ( remoteCount == 0 ) {
|
|
gi->serverRole = SERVER_STANDALONE;
|
|
}
|
|
}
|
|
|
|
*nPlayersHere = gi->nPlayers - remoteCount;
|
|
*nPlayersTotal = gi->nPlayers;
|
|
}
|
|
} /* checkServerRole */
|
|
|
|
static XP_U32
|
|
makeGameID( XW_UtilCtxt* XP_UNUSED_DBG(util) )
|
|
{
|
|
XP_U32 gameID = 0;
|
|
assertUtilOK( util );
|
|
while ( 0 == gameID ) {
|
|
/* High bit never set by XP_RANDOM() alone */
|
|
gameID = (XP_RANDOM() << 16) ^ XP_RANDOM();
|
|
/* But let's clear it -- set high-bit causes problems for existing
|
|
postgres DB where INTEGER is apparently a signed 32-bit. Recently
|
|
confirmed that working around that in the code that moves between
|
|
incoming web api calls and postgres would be much harder than using
|
|
8-char hex strings instead. But I'll leave the test change in
|
|
place. */
|
|
#ifdef HIGH_GAMEID_BITS
|
|
gameID |= 0x80000000;
|
|
#else
|
|
gameID &= ~0x80000000;
|
|
#endif
|
|
}
|
|
LOG_RETURNF( "%08X", gameID );
|
|
return gameID;
|
|
}
|
|
|
|
static void
|
|
timerChangeListener( XWEnv xwe, void* data, const XP_U32 gameID,
|
|
XP_S32 oldVal, XP_S32 newVal )
|
|
{
|
|
XWGame* game = (XWGame*)data;
|
|
XP_ASSERT( game->util->gameInfo->gameID == gameID );
|
|
XP_LOGFF( "(oldVal=%d, newVal=%d, id=%d)", oldVal, newVal, gameID );
|
|
dutil_onDupTimerChanged( util_getDevUtilCtxt( game->util, xwe ), xwe,
|
|
gameID, oldVal, newVal );
|
|
}
|
|
|
|
#ifdef XWFEATURE_RELAY
|
|
static void
|
|
onRoleChanged( XWEnv xwe, void* closure, XP_Bool amNowGuest )
|
|
{
|
|
XP_ASSERT( amNowGuest );
|
|
XWGame* game = (XWGame*)closure;
|
|
server_onRoleChanged( game->server, xwe, amNowGuest );
|
|
}
|
|
#endif
|
|
|
|
static void
|
|
setListeners( XWGame* game, const CommonPrefs* cp )
|
|
{
|
|
server_prefsChanged( game->server, cp );
|
|
board_prefsChanged( game->board, cp );
|
|
server_setTimerChangeListener( game->server, timerChangeListener, game );
|
|
}
|
|
|
|
static const DictionaryCtxt*
|
|
getDicts( const CurGameInfo* gi, XW_UtilCtxt* util, XWEnv xwe,
|
|
PlayerDicts* playerDicts )
|
|
{
|
|
const XP_UCHAR* isoCode = gi->isoCodeStr;
|
|
const DictionaryCtxt* result = util_getDict( util, xwe, isoCode, gi->dictName );
|
|
XP_MEMSET( playerDicts, 0, sizeof(*playerDicts) );
|
|
if ( !!result ) {
|
|
for ( int ii = 0; ii < gi->nPlayers; ++ii ) {
|
|
const LocalPlayer* lp = &gi->players[ii];
|
|
if ( lp->isLocal && !!lp->dictName && lp->dictName[0] ) {
|
|
playerDicts->dicts[ii] = util_getDict( util, xwe, isoCode,
|
|
lp->dictName );
|
|
}
|
|
}
|
|
}
|
|
return result;
|
|
}
|
|
|
|
static void
|
|
unrefDicts( XWEnv xwe, const DictionaryCtxt* dict, PlayerDicts* playerDicts )
|
|
{
|
|
if ( !!dict ) {
|
|
dict_unref( dict, xwe );
|
|
}
|
|
for ( int ii = 0; ii < VSIZE(playerDicts->dicts); ++ii ) {
|
|
const DictionaryCtxt* dict = playerDicts->dicts[ii];
|
|
if ( !!dict ) {
|
|
dict_unref( dict, xwe );
|
|
}
|
|
}
|
|
}
|
|
|
|
XP_Bool
|
|
game_makeNewGame( MPFORMAL XWEnv xwe, XWGame* game, CurGameInfo* gi,
|
|
const CommsAddrRec* selfAddr, const CommsAddrRec* hostAddr,
|
|
XW_UtilCtxt* util,
|
|
DrawCtx* draw, const CommonPrefs* cp,
|
|
const TransportProcs* procs
|
|
)
|
|
{
|
|
XP_ASSERT( gi == util->gameInfo ); /* if holds, remove gi param */
|
|
XP_U16 nPlayersHere = 0;
|
|
XP_U16 nPlayersTotal = 0;
|
|
checkServerRole( gi, &nPlayersHere, &nPlayersTotal );
|
|
assertUtilOK( util );
|
|
|
|
if ( 0 == gi->gameID ) {
|
|
gi->gameID = makeGameID( util );
|
|
}
|
|
game->created = dutil_getCurSeconds( util_getDevUtilCtxt( util, xwe ), xwe );
|
|
game->util = util;
|
|
|
|
PlayerDicts playerDicts;
|
|
const DictionaryCtxt* dict = getDicts( gi, util, xwe, &playerDicts );
|
|
XP_Bool success = !!dict;
|
|
|
|
if ( success ) {
|
|
XP_STRNCPY( gi->isoCodeStr, dict_getISOCode( dict ), VSIZE(gi->isoCodeStr) );
|
|
XP_ASSERT( !!gi->isoCodeStr[0] );
|
|
game->model = model_make( MPPARM(mpool) xwe, (DictionaryCtxt*)NULL,
|
|
NULL, util, gi->boardSize );
|
|
|
|
model_setDictionary( game->model, xwe, dict );
|
|
model_setPlayerDicts( game->model, xwe, &playerDicts );
|
|
|
|
if ( gi->serverRole != SERVER_STANDALONE ) {
|
|
game->comms = comms_make( MPPARM(mpool) xwe, util,
|
|
gi->serverRole != SERVER_ISCLIENT,
|
|
selfAddr, hostAddr, procs,
|
|
#ifdef XWFEATURE_RELAY
|
|
nPlayersHere, nPlayersTotal,
|
|
onRoleChanged, game,
|
|
#endif
|
|
gi->forceChannel
|
|
);
|
|
} else {
|
|
game->comms = (CommsCtxt*)NULL;
|
|
}
|
|
|
|
|
|
game->server = server_make( MPPARM(mpool) xwe, game->model,
|
|
game->comms, util );
|
|
game->board = board_make( MPPARM(mpool) xwe, game->model, game->server,
|
|
NULL, util );
|
|
board_setCallbacks( game->board, xwe );
|
|
|
|
board_setDraw( game->board, xwe, draw );
|
|
setListeners( game, cp );
|
|
}
|
|
|
|
unrefDicts( xwe, dict, &playerDicts );
|
|
|
|
return success;
|
|
} /* game_makeNewGame */
|
|
|
|
XP_Bool
|
|
game_makeRematch( const XWGame* oldGame, XWEnv xwe, XW_UtilCtxt* newUtil,
|
|
const CommonPrefs* newCp, const TransportProcs* procs,
|
|
XWGame* newGame, const XP_UCHAR* newName, NewOrder* nop )
|
|
{
|
|
XP_Bool success = XP_FALSE;
|
|
|
|
RematchInfo* rip;
|
|
if ( server_getRematchInfo( oldGame->server, newUtil,
|
|
makeGameID( newUtil ), nop, &rip ) ) {
|
|
CommsAddrRec* selfAddrP = NULL;
|
|
CommsAddrRec selfAddr;
|
|
if ( !!oldGame->comms ) {
|
|
comms_getSelfAddr( oldGame->comms, &selfAddr );
|
|
selfAddrP = &selfAddr;
|
|
}
|
|
|
|
if ( game_makeNewGame( MPPARM(newUtil->mpool) xwe, newGame,
|
|
newUtil->gameInfo, selfAddrP, (CommsAddrRec*)NULL,
|
|
newUtil, (DrawCtx*)NULL, newCp, procs ) ) {
|
|
if ( !!newGame->comms ) {
|
|
server_setRematchOrder( newGame->server, rip );
|
|
|
|
const CurGameInfo* newGI = newUtil->gameInfo;
|
|
for ( int ii = 0; ; ++ii ) {
|
|
CommsAddrRec guestAddr;
|
|
XP_U16 nPlayersH;
|
|
if ( !server_ri_getAddr( rip, ii, &guestAddr, &nPlayersH ) ) {
|
|
break;
|
|
}
|
|
XP_ASSERT( !comms_addrsAreSame( newGame->comms, &guestAddr,
|
|
&selfAddr ) );
|
|
|
|
NetLaunchInfo nli;
|
|
nli_init( &nli, newGI, selfAddrP, nPlayersH, ii + 1 );
|
|
if ( !!newName ) {
|
|
nli_setGameName( &nli, newName );
|
|
}
|
|
LOGNLI( &nli );
|
|
comms_invite( newGame->comms, xwe, &nli, &guestAddr, XP_TRUE );
|
|
}
|
|
}
|
|
success = XP_TRUE;
|
|
}
|
|
server_disposeRematchInfo( oldGame->server, &rip );
|
|
}
|
|
XP_LOGFF( "=> %s; game with gid %08X rematched to create game "
|
|
"with gid %08X",
|
|
boolToStr(success), oldGame->util->gameInfo->gameID,
|
|
newUtil->gameInfo->gameID );
|
|
return success;
|
|
}
|
|
|
|
#ifdef XWFEATURE_CHANGEDICT
|
|
void
|
|
game_changeDict( MPFORMAL XWGame* game, XWEnv xwe, CurGameInfo* gi, DictionaryCtxt* dict )
|
|
{
|
|
model_setDictionary( game->model, xwe, dict );
|
|
gi_setDict( MPPARM(mpool) gi, dict );
|
|
server_resetEngines( game->server );
|
|
}
|
|
#endif
|
|
|
|
XP_Bool
|
|
game_makeFromStream( MPFORMAL XWEnv xwe, XWStreamCtxt* stream,
|
|
XWGame* game, CurGameInfo* gi,
|
|
XW_UtilCtxt* util, DrawCtx* draw, CommonPrefs* cp,
|
|
const TransportProcs* procs )
|
|
{
|
|
XP_ASSERT( NULL == util || gi == util->gameInfo );
|
|
XP_Bool success = XP_FALSE;
|
|
XP_Bool hasComms;
|
|
XP_U8 strVersion = stream_getU8( stream );
|
|
XP_LOGFF( "strVersion = 0x%x", (XP_U16)strVersion );
|
|
|
|
if ( strVersion > CUR_STREAM_VERS ) {
|
|
XP_LOGFF( "aborting; stream version too new (%d > %d)!",
|
|
strVersion, CUR_STREAM_VERS );
|
|
} else {
|
|
do { /* do..while so can break */
|
|
stream_setVersion( stream, strVersion );
|
|
|
|
gi_readFromStream( MPPARM(mpool) stream, gi );
|
|
if ( !game ) {
|
|
success = XP_TRUE;
|
|
break;
|
|
} else if ( stream_getSize(stream) == 0 ) {
|
|
XP_LOGFF( "gi was all we got; failing." );
|
|
break;
|
|
}
|
|
game->util = util;
|
|
game->created = strVersion < STREAM_VERS_GICREATED
|
|
? 0 : stream_getU32( stream );
|
|
|
|
PlayerDicts playerDicts;
|
|
const DictionaryCtxt* dict = getDicts( gi, util, xwe, &playerDicts );
|
|
if ( !dict ) {
|
|
break;
|
|
}
|
|
|
|
/* Previous stream versions didn't save anything if built
|
|
* standalone. Now we always save something. But we need to know
|
|
* if the previous version didn't save. PREV_WAS_STANDALONE_ONLY
|
|
* tells us that.
|
|
*/
|
|
hasComms = XP_FALSE;
|
|
if ( STREAM_VERS_ALWAYS_MULTI <= strVersion /* new stream */
|
|
#ifndef PREV_WAS_STANDALONE_ONLY
|
|
|| XP_TRUE /* old, but saved this anyway */
|
|
#endif
|
|
) {
|
|
if ( strVersion < STREAM_VERS_GICREATED ) {
|
|
hasComms = stream_getU8( stream );
|
|
} else {
|
|
XP_U8 flags = stream_getU8( stream );
|
|
hasComms = flags & FLAG_HASCOMMS;
|
|
}
|
|
}
|
|
|
|
XP_ASSERT( hasComms == (SERVER_STANDALONE != gi->serverRole) );
|
|
if ( hasComms ) {
|
|
game->comms = comms_makeFromStream( MPPARM(mpool) xwe, stream, util,
|
|
gi->serverRole != SERVER_ISCLIENT,
|
|
procs,
|
|
#ifdef XWFEATURE_RELAY
|
|
onRoleChanged, game,
|
|
#endif
|
|
gi->forceChannel );
|
|
} else {
|
|
XP_ASSERT( NULL == game->comms );
|
|
game->comms = NULL;
|
|
}
|
|
|
|
game->model = model_makeFromStream( MPPARM(mpool) xwe, stream, dict,
|
|
&playerDicts, util );
|
|
|
|
game->server = server_makeFromStream( MPPARM(mpool) xwe, stream,
|
|
game->model, game->comms,
|
|
util, gi->nPlayers );
|
|
|
|
game->board = board_makeFromStream( MPPARM(mpool) xwe, stream,
|
|
game->model, game->server,
|
|
NULL, util, gi->nPlayers );
|
|
setListeners( game, cp );
|
|
board_setDraw( game->board, xwe, draw );
|
|
success = XP_TRUE;
|
|
unrefDicts( xwe, dict, &playerDicts );
|
|
} while( XP_FALSE );
|
|
}
|
|
|
|
if ( success && !!game && !!game->comms ) {
|
|
XP_ASSERT( comms_getIsHost(game->comms) == server_getIsHost(game->server) );
|
|
|
|
#ifdef XWFEATURE_KNOWNPLAYERS
|
|
const XP_U32 created = game->created;
|
|
if ( 0 != created
|
|
&& server_getGameIsConnected( game->server ) ) {
|
|
comms_gatherPlayers( game->comms, xwe, created );
|
|
}
|
|
#endif
|
|
}
|
|
|
|
return success;
|
|
} /* game_makeFromStream */
|
|
|
|
/* This is a gross hack. Fix it someday. */
|
|
static void
|
|
runServer( ServerCtxt* server, XWEnv xwe )
|
|
{
|
|
for ( int ii = 0; ii < 5; ++ii ) {
|
|
(void)server_do( server, xwe );
|
|
}
|
|
}
|
|
|
|
XP_Bool
|
|
game_makeFromInvite( XWGame* newGame, XWEnv xwe, const NetLaunchInfo* nli,
|
|
const CommsAddrRec* selfAddr, XW_UtilCtxt* util,
|
|
DrawCtx* draw, CommonPrefs* cp, const TransportProcs* procs )
|
|
{
|
|
LOG_FUNC();
|
|
XP_U32 gameID = nli->gameID;
|
|
XP_U8 forceChannel = nli->forceChannel;
|
|
XW_DUtilCtxt* duc = util_getDevUtilCtxt( util, xwe );
|
|
XP_Bool success = !dutil_haveGame( duc, xwe, gameID, forceChannel );
|
|
if ( success ) {
|
|
CurGameInfo* gi = util->gameInfo;
|
|
XP_ASSERT( !!gi );
|
|
nliToGI( nli, xwe, util, gi );
|
|
|
|
CommsAddrRec hostAddr;
|
|
nli_makeAddrRec( nli, &hostAddr );
|
|
|
|
success = game_makeNewGame( MPPARM(util->mpool) xwe, newGame,
|
|
gi, selfAddr, &hostAddr, util,
|
|
draw, cp, procs );
|
|
if ( success && server_initClientConnection( newGame->server, xwe ) ) {
|
|
runServer( newGame->server, xwe );
|
|
}
|
|
}
|
|
LOG_RETURNF( "%s", boolToStr(success) );
|
|
return success;
|
|
}
|
|
|
|
void
|
|
game_saveToStream( const XWGame* game, const CurGameInfo* gi,
|
|
XWStreamCtxt* stream, XP_U16 saveToken )
|
|
{
|
|
XP_ASSERT( gi_equal( gi, game->util->gameInfo ) );
|
|
stream_putU8( stream, CUR_STREAM_VERS );
|
|
stream_setVersion( stream, CUR_STREAM_VERS );
|
|
|
|
gi_writeToStream( stream, gi );
|
|
|
|
if ( !!game ) {
|
|
const XP_U32 created = game->created;
|
|
stream_putU32( stream, created );
|
|
XP_ASSERT( 0 != saveToken );
|
|
|
|
XP_U8 flags = NULL == game->comms ? 0 : FLAG_HASCOMMS;
|
|
stream_putU8( stream, flags );
|
|
if ( NULL != game->comms ) {
|
|
comms_writeToStream( game->comms, stream, saveToken );
|
|
}
|
|
|
|
model_writeToStream( game->model, stream );
|
|
server_writeToStream( game->server, stream );
|
|
board_writeToStream( game->board, stream );
|
|
}
|
|
} /* game_saveToStream */
|
|
|
|
void
|
|
game_saveSucceeded( const XWGame* game, XWEnv xwe, XP_U16 saveToken )
|
|
{
|
|
if ( !!game->comms ) {
|
|
comms_saveSucceeded( game->comms, xwe, saveToken );
|
|
}
|
|
}
|
|
|
|
XP_Bool
|
|
game_receiveMessage( XWGame* game, XWEnv xwe, XWStreamCtxt* stream,
|
|
const CommsAddrRec* retAddr )
|
|
{
|
|
ServerCtxt* server = game->server;
|
|
CommsMsgState commsState;
|
|
XP_Bool result = NULL != game->comms;
|
|
if ( result ) {
|
|
result = comms_checkIncomingStream( game->comms, xwe, stream, retAddr,
|
|
&commsState );
|
|
} else {
|
|
XP_LOGFF( "ERROR: comms NULL!" );
|
|
}
|
|
if ( result ) {
|
|
(void)server_do( server, xwe );
|
|
|
|
result = server_receiveMessage( server, xwe, stream );
|
|
}
|
|
comms_msgProcessed( game->comms, xwe, &commsState, !result );
|
|
|
|
if ( result ) {
|
|
/* in case MORE work's pending. Multiple calls are required in at
|
|
least one case, where I'm a host handling client registration *AND*
|
|
I'm a robot. Only one server_do and I'll never make that first
|
|
robot move. That's because comms can't detect a duplicate initial
|
|
packet (in validateInitialMessage()). */
|
|
runServer( server, xwe );
|
|
}
|
|
|
|
LOG_RETURNF( "%s", boolToStr(result) );
|
|
return result;
|
|
}
|
|
|
|
void
|
|
game_getState( const XWGame* game, XWEnv xwe, GameStateInfo* gsi )
|
|
{
|
|
const ServerCtxt* server = game->server;
|
|
BoardCtxt* board = game->board;
|
|
|
|
XP_Bool gameOver = server_getGameIsOver( server );
|
|
gsi->curTurnSelected = board_curTurnSelected( board );
|
|
gsi->trayVisState = board_getTrayVisState( board );
|
|
gsi->visTileCount = board_visTileCount( board );
|
|
gsi->canHint = !gameOver && board_canHint( board );
|
|
gsi->canUndo = model_canUndo( game->model );
|
|
gsi->canRedo = board_canTogglePending( board );
|
|
gsi->inTrade = board_inTrade( board, &gsi->tradeTilesSelected );
|
|
gsi->canChat = !!game->comms && comms_canChat( game->comms );
|
|
gsi->canShuffle = board_canShuffle( board );
|
|
gsi->canHideRack = board_canHideRack( board );
|
|
gsi->canTrade = board_canTrade( board, xwe );
|
|
gsi->nPendingMessages = !!game->comms ?
|
|
comms_countPendingPackets(game->comms, NULL) : 0;
|
|
gsi->canPause = server_canPause( server );
|
|
gsi->canUnpause = server_canUnpause( server );
|
|
}
|
|
|
|
void
|
|
game_summarize( const XWGame* game, const CurGameInfo* gi, GameSummary* summary )
|
|
{
|
|
XP_MEMSET( summary, 0, sizeof(*summary) );
|
|
ServerCtxt* server = game->server;
|
|
summary->turn = server_getCurrentTurn( server, &summary->turnIsLocal );
|
|
summary->lastMoveTime = server_getLastMoveTime(server);
|
|
XP_STRNCPY( summary->isoCodeStr, gi->isoCodeStr, VSIZE(summary->isoCodeStr)-1 );
|
|
summary->gameOver = server_getGameIsOver( server );
|
|
summary->nMoves = model_getNMoves( game->model );
|
|
summary->dupTimerExpires = server_getDupTimerExpires( server );
|
|
summary->canRematch = server_canRematch( server, &summary->canOfferRO );
|
|
|
|
for ( int ii = 0; ii < gi->nPlayers; ++ii ) {
|
|
const LocalPlayer* lp = &gi->players[ii];
|
|
if ( LP_IS_ROBOT(lp) || !LP_IS_LOCAL(lp) ) {
|
|
if ( '\0' != summary->opponents[0] ) {
|
|
XP_STRCAT( summary->opponents, ", " );
|
|
}
|
|
XP_STRCAT( summary->opponents, lp->name );
|
|
}
|
|
}
|
|
if ( !!game->comms ) {
|
|
summary->missingPlayers = server_getMissingPlayers( server );
|
|
summary->nPacketsPending =
|
|
comms_countPendingPackets( game->comms, &summary->quashed );
|
|
summary->channelNo = gi->forceChannel;
|
|
}
|
|
}
|
|
|
|
XP_Bool
|
|
game_getIsHost( const XWGame* game )
|
|
{
|
|
XP_Bool result = comms_getIsHost( game->comms );
|
|
return result;
|
|
}
|
|
|
|
void
|
|
game_dispose( XWGame* game, XWEnv xwe )
|
|
{
|
|
#ifdef XWFEATURE_KNOWNPLAYERS
|
|
const XP_U32 created = game->created;
|
|
if ( !!game->comms && 0 != created
|
|
&& server_getGameIsConnected( game->server ) ) {
|
|
comms_gatherPlayers( game->comms, xwe, created );
|
|
}
|
|
#endif
|
|
|
|
/* The board should be reused!!! PENDING(ehouse) */
|
|
if ( !!game->board ) {
|
|
board_destroy( game->board, xwe, XP_TRUE );
|
|
game->board = NULL;
|
|
}
|
|
|
|
if ( !!game->comms ) {
|
|
comms_stop( game->comms
|
|
#ifdef XWFEATURE_RELAY
|
|
, xwe
|
|
#endif
|
|
);
|
|
comms_destroy( game->comms, xwe );
|
|
game->comms = NULL;
|
|
}
|
|
if ( !!game->model ) {
|
|
model_destroy( game->model, xwe );
|
|
game->model = NULL;
|
|
}
|
|
if ( !!game->server ) {
|
|
server_destroy( game->server );
|
|
game->server = NULL;
|
|
}
|
|
} /* game_dispose */
|
|
|
|
static void
|
|
disposePlayerInfoInt( MPFORMAL CurGameInfo* gi )
|
|
{
|
|
for ( int ii = 0; ii < VSIZE(gi->players); ++ii ) {
|
|
LocalPlayer* lp = &gi->players[ii];
|
|
XP_FREEP( mpool, &lp->name );
|
|
XP_FREEP( mpool, &lp->password );
|
|
XP_FREEP( mpool, &lp->dictName );
|
|
}
|
|
} /* disposePlayerInfoInt */
|
|
|
|
void
|
|
gi_disposePlayerInfo( MPFORMAL CurGameInfo* gi )
|
|
{
|
|
disposePlayerInfoInt( MPPARM(mpool) gi );
|
|
|
|
XP_FREEP( mpool, &gi->dictName );
|
|
} /* gi_disposePlayerInfo */
|
|
|
|
void
|
|
gi_copy( MPFORMAL CurGameInfo* destGI, const CurGameInfo* srcGI )
|
|
{
|
|
replaceStringIfDifferent( mpool, &destGI->dictName,
|
|
srcGI->dictName );
|
|
XP_STRNCPY( destGI->isoCodeStr, srcGI->isoCodeStr, VSIZE(destGI->isoCodeStr)-1 );
|
|
destGI->gameID = srcGI->gameID;
|
|
destGI->gameSeconds = srcGI->gameSeconds;
|
|
destGI->nPlayers = (XP_U8)srcGI->nPlayers;
|
|
XP_U16 nPlayers = srcGI->nPlayers;
|
|
destGI->boardSize = (XP_U8)srcGI->boardSize;
|
|
destGI->traySize = srcGI->traySize;
|
|
destGI->bingoMin = srcGI->bingoMin;
|
|
destGI->serverRole = srcGI->serverRole;
|
|
|
|
destGI->hintsNotAllowed = srcGI->hintsNotAllowed;
|
|
destGI->timerEnabled = srcGI->timerEnabled;
|
|
destGI->phoniesAction = srcGI->phoniesAction;
|
|
destGI->allowPickTiles = srcGI->allowPickTiles;
|
|
destGI->forceChannel = srcGI->forceChannel;
|
|
destGI->inDuplicateMode = srcGI->inDuplicateMode;
|
|
XP_LOGFF( "copied forceChannel: %d; inDuplicateMode: %d",
|
|
destGI->forceChannel, destGI->inDuplicateMode );
|
|
|
|
const LocalPlayer* srcPl;
|
|
LocalPlayer* destPl;
|
|
int ii;
|
|
for ( srcPl = srcGI->players, destPl = destGI->players, ii = 0;
|
|
ii < nPlayers; ++srcPl, ++destPl, ++ii ) {
|
|
|
|
replaceStringIfDifferent( mpool, &destPl->name, srcPl->name );
|
|
replaceStringIfDifferent( mpool, &destPl->password,
|
|
srcPl->password );
|
|
replaceStringIfDifferent( mpool, &destPl->dictName,
|
|
srcPl->dictName );
|
|
destPl->secondsUsed = srcPl->secondsUsed;
|
|
destPl->robotIQ = srcPl->robotIQ;
|
|
destPl->isLocal = srcPl->isLocal;
|
|
}
|
|
} /* gi_copy */
|
|
|
|
#ifdef DEBUG
|
|
static XP_Bool
|
|
strEq( const XP_UCHAR* str1, const XP_UCHAR* str2 )
|
|
{
|
|
if ( NULL == str1 ) {
|
|
str1 = "";
|
|
}
|
|
if ( NULL == str2 ) {
|
|
str2 = "";
|
|
}
|
|
return 0 == XP_STRCMP(str1, str2);
|
|
}
|
|
|
|
XP_Bool
|
|
gi_equal( const CurGameInfo* gi1, const CurGameInfo* gi2 )
|
|
{
|
|
XP_Bool equal = XP_FALSE;
|
|
int ii;
|
|
for ( ii = 0; ; ++ii ) {
|
|
switch ( ii ) {
|
|
case 0:
|
|
equal = gi1->gameID == gi2->gameID;
|
|
break;
|
|
case 1:
|
|
equal = gi1->gameSeconds == gi2->gameSeconds;
|
|
break;
|
|
case 2:
|
|
equal = gi1->nPlayers == gi2->nPlayers;
|
|
break;
|
|
case 3:
|
|
equal = gi1->boardSize == gi2->boardSize;
|
|
break;
|
|
case 4:
|
|
equal = gi1->traySize == gi2->traySize;
|
|
break;
|
|
case 5:
|
|
equal = gi1->bingoMin == gi2->bingoMin;
|
|
break;
|
|
case 6:
|
|
equal = gi1->forceChannel == gi2->forceChannel;
|
|
break;
|
|
case 7:
|
|
equal = gi1->serverRole == gi2->serverRole;
|
|
break;
|
|
case 8:
|
|
equal = gi1->hintsNotAllowed == gi2->hintsNotAllowed;
|
|
break;
|
|
case 9:
|
|
equal = gi1->timerEnabled == gi2->timerEnabled;
|
|
break;
|
|
case 10:
|
|
equal = gi1->allowPickTiles == gi2->allowPickTiles;
|
|
break;
|
|
case 11:
|
|
equal = gi1->allowHintRect == gi2->allowHintRect;
|
|
break;
|
|
case 12:
|
|
equal = gi1->inDuplicateMode == gi2->inDuplicateMode;
|
|
break;
|
|
case 13:
|
|
equal = gi1->phoniesAction == gi2->phoniesAction;
|
|
break;
|
|
case 14:
|
|
equal = gi1->confirmBTConnect == gi2->confirmBTConnect;
|
|
break;
|
|
case 15:
|
|
equal = strEq( gi1->dictName, gi2->dictName );
|
|
break;
|
|
case 16:
|
|
equal = strEq( gi1->isoCodeStr, gi2->isoCodeStr );
|
|
break;
|
|
case 17:
|
|
for ( int jj = 0; equal && jj < gi1->nPlayers; ++jj ) {
|
|
const LocalPlayer* lp1 = &gi1->players[jj];
|
|
const LocalPlayer* lp2 = &gi2->players[jj];
|
|
equal = strEq( lp1->name, lp2->name )
|
|
&& strEq( lp1->password, lp2->password )
|
|
&& strEq( lp1->dictName, lp2->dictName )
|
|
&& lp1->secondsUsed == lp2->secondsUsed
|
|
&& lp1->isLocal == lp2->isLocal
|
|
&& lp1->robotIQ == lp2->robotIQ
|
|
;
|
|
}
|
|
break;
|
|
default:
|
|
goto done;
|
|
break;
|
|
}
|
|
if ( !equal ) {
|
|
break;
|
|
}
|
|
}
|
|
done:
|
|
if ( !equal ) {
|
|
XP_LOGFF( "exited when ii = %d", ii );
|
|
LOGGI( gi1, "gi1" );
|
|
LOGGI( gi2, "gi2" );
|
|
}
|
|
|
|
return equal;
|
|
}
|
|
#endif
|
|
|
|
|
|
void
|
|
gi_setNPlayers( CurGameInfo* gi, XWEnv xwe, XW_UtilCtxt* util,
|
|
XP_U16 nTotal, XP_U16 nHere )
|
|
{
|
|
LOGGI( gi, "before" );
|
|
XP_ASSERT( nTotal <= MAX_NUM_PLAYERS );
|
|
XP_ASSERT( nHere < nTotal );
|
|
|
|
gi->nPlayers = nTotal;
|
|
|
|
XP_U16 curLocal = 0;
|
|
for ( XP_U16 ii = 0; ii < nTotal; ++ii ) {
|
|
if ( gi->players[ii].isLocal ) {
|
|
++curLocal;
|
|
}
|
|
}
|
|
|
|
if ( nHere != curLocal ) {
|
|
/* This will happen when a device has more than one player. Not sure I
|
|
handle that correctly, but don't assert for now. */
|
|
XP_LOGFF( "nHere: %d; curLocal: %d; a problem?", nHere, curLocal );
|
|
for ( XP_U16 ii = 0; ii < nTotal; ++ii ) {
|
|
if ( !gi->players[ii].isLocal ) {
|
|
gi->players[ii].isLocal = XP_TRUE;
|
|
XP_LOGFF( "making player #%d local when wasn't before", ii );
|
|
++curLocal;
|
|
XP_ASSERT( curLocal <= nHere );
|
|
if ( curLocal == nHere ) {
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
for ( XP_U16 ii = 0; ii < nTotal; ++ii ) {
|
|
LocalPlayer* lp = &gi->players[ii];
|
|
if ( !lp->name || !lp->name[0] ) {
|
|
XP_UCHAR name[32];
|
|
XP_U16 len = VSIZE(name);
|
|
dutil_getUsername( util_getDevUtilCtxt( util, xwe ),
|
|
xwe, ii, LP_IS_LOCAL(lp),
|
|
LP_IS_ROBOT(lp), name, &len );
|
|
replaceStringIfDifferent( util->mpool, &lp->name, name );
|
|
}
|
|
}
|
|
|
|
LOGGI( gi, "after" );
|
|
}
|
|
|
|
XP_U16
|
|
gi_countLocalPlayers( const CurGameInfo* gi, XP_Bool humanOnly )
|
|
{
|
|
XP_U16 count = 0;
|
|
XP_U16 nPlayers = gi->nPlayers;
|
|
const LocalPlayer* lp = gi->players;
|
|
while ( nPlayers-- ) {
|
|
if ( lp->isLocal ) {
|
|
if ( humanOnly && LP_IS_ROBOT(lp) ) {
|
|
// skip
|
|
} else {
|
|
++count;
|
|
}
|
|
}
|
|
++lp;
|
|
}
|
|
return count;
|
|
} /* gi_countLocalPlayers */
|
|
|
|
void
|
|
gi_readFromStream( MPFORMAL XWStreamCtxt* stream, CurGameInfo* gi )
|
|
{
|
|
LocalPlayer* pl;
|
|
XP_U16 ii;
|
|
XP_UCHAR* str;
|
|
XP_U16 strVersion = stream_getVersion( stream );
|
|
XP_U16 nColsNBits;
|
|
XP_ASSERT( 0 < strVersion );
|
|
#ifdef STREAM_VERS_BIGBOARD
|
|
nColsNBits = STREAM_VERS_BIGBOARD > strVersion ? NUMCOLS_NBITS_4
|
|
: NUMCOLS_NBITS_5;
|
|
#else
|
|
nColsNBits = NUMCOLS_NBITS_4;
|
|
#endif
|
|
|
|
str = stringFromStream( mpool, stream );
|
|
replaceStringIfDifferent( mpool, &gi->dictName, str );
|
|
XP_FREEP( mpool, &str );
|
|
|
|
gi->nPlayers = (XP_U8)stream_getBits( stream, NPLAYERS_NBITS );
|
|
gi->boardSize = (XP_U8)stream_getBits( stream, nColsNBits );
|
|
if ( STREAM_VERS_NINETILES <= strVersion ) {
|
|
gi->traySize = (XP_U8)stream_getBits( stream, NTILES_NBITS_9 );
|
|
gi->bingoMin = (XP_U8)stream_getBits( stream, NTILES_NBITS_9 );
|
|
} else {
|
|
gi->traySize = gi->bingoMin = 7;
|
|
}
|
|
gi->serverRole = (DeviceRole)stream_getBits( stream, 2 );
|
|
/* XP_LOGF( "%s: read serverRole of %d", __func__, gi->serverRole ); */
|
|
gi->hintsNotAllowed = stream_getBits( stream, 1 );
|
|
if ( strVersion < STREAM_VERS_ROBOTIQ ) {
|
|
(void)stream_getBits( stream, 2 );
|
|
}
|
|
gi->phoniesAction = (XWPhoniesChoice)stream_getBits( stream, 2 );
|
|
gi->timerEnabled = stream_getBits( stream, 1 );
|
|
|
|
gi->inDuplicateMode = strVersion >= STREAM_VERS_DUPLICATE
|
|
? stream_getBits( stream, 1 )
|
|
: XP_FALSE;
|
|
if ( strVersion >= STREAM_VERS_41B4 ) {
|
|
gi->allowPickTiles = stream_getBits( stream, 1 );
|
|
gi->allowHintRect = stream_getBits( stream, 1 );
|
|
} else {
|
|
gi->allowPickTiles = XP_FALSE;
|
|
gi->allowHintRect = XP_FALSE;
|
|
}
|
|
|
|
if ( strVersion >= STREAM_VERS_BLUETOOTH ) {
|
|
gi->confirmBTConnect = stream_getBits( stream, 1 );
|
|
} else {
|
|
gi->confirmBTConnect = XP_TRUE; /* safe given all the 650s out there. */
|
|
}
|
|
|
|
if ( STREAM_VERS_MULTIADDR <= strVersion ) {
|
|
gi->forceChannel = stream_getBits( stream, 2 );
|
|
}
|
|
gi->gameID = strVersion < STREAM_VERS_BLUETOOTH2 ?
|
|
stream_getU16( stream ) : stream_getU32( stream );
|
|
// XP_LOGFF( "read forceChannel: %d for gid %X", gi->forceChannel, gi->gameID );
|
|
|
|
if ( STREAM_VERS_GI_ISO <= strVersion ) {
|
|
stringFromStreamHere( stream, gi->isoCodeStr, VSIZE(gi->isoCodeStr) );
|
|
} else if ( STREAM_VERS_DICTLANG <= strVersion ) {
|
|
XP_LangCode dictLang = stream_getU8( stream );
|
|
const XP_UCHAR* isoCode = lcToLocale( dictLang );
|
|
XP_ASSERT( !!isoCode );
|
|
XP_STRNCPY( gi->isoCodeStr, isoCode, VSIZE(gi->isoCodeStr) );
|
|
XP_LOGFF( "upgrading; faked isoCode: %s", gi->isoCodeStr );
|
|
}
|
|
|
|
if ( gi->timerEnabled || strVersion >= STREAM_VERS_GAMESECONDS ) {
|
|
gi->gameSeconds = stream_getU16( stream );
|
|
}
|
|
|
|
for ( pl = gi->players, ii = 0; ii < gi->nPlayers; ++pl, ++ii ) {
|
|
str = stringFromStream( mpool, stream );
|
|
replaceStringIfDifferent( mpool, &pl->name, str );
|
|
XP_FREEP( mpool, &str );
|
|
|
|
str = stringFromStream( mpool, stream );
|
|
replaceStringIfDifferent( mpool, &pl->password, str );
|
|
XP_FREEP( mpool, &str );
|
|
|
|
if ( strVersion >= STREAM_VERS_PLAYERDICTS ) {
|
|
str = stringFromStream( mpool, stream );
|
|
replaceStringIfDifferent( mpool, &pl->dictName, str );
|
|
XP_FREEP( mpool, &str );
|
|
}
|
|
|
|
pl->secondsUsed = stream_getU16( stream );
|
|
pl->robotIQ = ( strVersion < STREAM_VERS_ROBOTIQ )
|
|
? (XP_U8)stream_getBits( stream, 1 )
|
|
: stream_getU8( stream );
|
|
pl->isLocal = stream_getBits( stream, 1 );
|
|
}
|
|
} /* gi_readFromStream */
|
|
|
|
void
|
|
gi_writeToStream( XWStreamCtxt* stream, const CurGameInfo* gi )
|
|
{
|
|
XP_U16 nColsNBits;
|
|
#ifdef STREAM_VERS_BIGBOARD
|
|
XP_U16 strVersion = stream_getVersion( stream );
|
|
/* XP_LOGF( "%s: strVersion = 0x%x", __func__, strVersion ); */
|
|
XP_ASSERT( STREAM_SAVE_PREVWORDS <= strVersion );
|
|
nColsNBits = STREAM_VERS_BIGBOARD > strVersion ? NUMCOLS_NBITS_4
|
|
: NUMCOLS_NBITS_5;
|
|
#else
|
|
nColsNBits = NUMCOLS_NBITS_4;
|
|
#endif
|
|
|
|
stringToStream( stream, gi->dictName );
|
|
|
|
stream_putBits( stream, NPLAYERS_NBITS, gi->nPlayers );
|
|
stream_putBits( stream, nColsNBits, gi->boardSize );
|
|
|
|
if ( STREAM_VERS_NINETILES <= strVersion ) {
|
|
XP_ASSERT( 0 < gi->traySize );
|
|
stream_putBits( stream, NTILES_NBITS_9, gi->traySize );
|
|
stream_putBits( stream, NTILES_NBITS_9, gi->bingoMin );
|
|
} else {
|
|
XP_LOGFF( "strVersion: %d so not writing traySize", strVersion );
|
|
}
|
|
stream_putBits( stream, 2, gi->serverRole );
|
|
stream_putBits( stream, 1, gi->hintsNotAllowed );
|
|
stream_putBits( stream, 2, gi->phoniesAction );
|
|
stream_putBits( stream, 1, gi->timerEnabled );
|
|
stream_putBits( stream, 1, gi->inDuplicateMode );
|
|
stream_putBits( stream, 1, gi->allowPickTiles );
|
|
stream_putBits( stream, 1, gi->allowHintRect );
|
|
stream_putBits( stream, 1, gi->confirmBTConnect );
|
|
stream_putBits( stream, 2, gi->forceChannel );
|
|
// XP_LOGFF( "wrote forceChannel: %d for gid %X", gi->forceChannel, gi->gameID );
|
|
|
|
if ( 0 ) {
|
|
#ifdef STREAM_VERS_BIGBOARD
|
|
} else if ( STREAM_VERS_BIGBOARD <= strVersion ) {
|
|
stream_putU32( stream, gi->gameID );
|
|
#endif
|
|
} else {
|
|
stream_putU16( stream, gi->gameID );
|
|
}
|
|
|
|
if ( STREAM_VERS_GI_ISO <= strVersion ) {
|
|
stringToStream( stream, gi->isoCodeStr );
|
|
} else {
|
|
XP_LangCode code;
|
|
if ( haveLocaleToLc( gi->isoCodeStr, &code ) ) {
|
|
stream_putU8( stream, code );
|
|
} else {
|
|
XP_ASSERT( 0 );
|
|
}
|
|
}
|
|
stream_putU16( stream, gi->gameSeconds );
|
|
|
|
int ii;
|
|
const LocalPlayer* pl;
|
|
for ( pl = gi->players, ii = 0; ii < gi->nPlayers; ++pl, ++ii ) {
|
|
stringToStream( stream, pl->name );
|
|
stringToStream( stream, pl->password );
|
|
stringToStream( stream, pl->dictName );
|
|
stream_putU16( stream, pl->secondsUsed );
|
|
stream_putU8( stream, pl->robotIQ );
|
|
stream_putBits( stream, 1, pl->isLocal );
|
|
}
|
|
} /* gi_writeToStream */
|
|
|
|
#ifdef XWFEATURE_CHANGEDICT
|
|
static void
|
|
gi_setDict( MPFORMAL CurGameInfo* gi, const DictionaryCtxt* dict )
|
|
{
|
|
XP_U16 ii;
|
|
const XP_UCHAR* name = dict_getName( dict );
|
|
replaceStringIfDifferent( mpool, &gi->dictName, name );
|
|
for ( ii = 0; ii < gi->nPlayers; ++ii ) {
|
|
const LocalPlayer* pl = &gi->players[ii];
|
|
XP_FREEP( mpool, &pl->dictName );
|
|
}
|
|
}
|
|
#endif
|
|
|
|
XP_Bool
|
|
player_hasPasswd( const LocalPlayer* player )
|
|
{
|
|
XP_UCHAR* password = player->password;
|
|
/* XP_ASSERT( player->isLocal ); */
|
|
return !!password && *password != '\0';
|
|
} /* player_hasPasswd */
|
|
|
|
XP_Bool
|
|
player_passwordMatches( const LocalPlayer* player, const XP_UCHAR* buf )
|
|
{
|
|
XP_ASSERT( player->isLocal );
|
|
|
|
return 0 == XP_STRCMP( player->password, (XP_UCHAR*)buf );
|
|
} /* player_passwordMatches */
|
|
|
|
XP_U16
|
|
player_timePenalty( CurGameInfo* gi, XP_U16 playerNum )
|
|
{
|
|
XP_S16 seconds = (gi->gameSeconds / gi->nPlayers);
|
|
LocalPlayer* player = gi->players + playerNum;
|
|
XP_U16 result = 0;
|
|
|
|
seconds -= player->secondsUsed;
|
|
if ( seconds < 0 ) {
|
|
seconds = -seconds;
|
|
seconds += 59;
|
|
result = (seconds/60) * 10;
|
|
}
|
|
return result;
|
|
} /* player_timePenalty */
|
|
|
|
#ifdef DEBUG
|
|
void
|
|
game_logGI( const CurGameInfo* gi, const char* msg, const char* func, int line )
|
|
{
|
|
XP_LOGFF( "msg: %s from %s() line %d; gameID: %X", msg, func, line,
|
|
!!gi ? gi->gameID:0 );
|
|
if ( !!gi ) {
|
|
XP_LOGF( " nPlayers: %d", gi->nPlayers );
|
|
for ( XP_U16 ii = 0; ii < gi->nPlayers; ++ii ) {
|
|
const LocalPlayer* lp = &gi->players[ii];
|
|
XP_LOGF( " player[%d]: local: %s; robotIQ: %d; name: %s; dict: %s; pwd: %s", ii,
|
|
boolToStr(lp->isLocal), lp->robotIQ, lp->name, lp->dictName, lp->password );
|
|
}
|
|
XP_LOGF( " forceChannel: %d", gi->forceChannel );
|
|
XP_LOGF( " serverRole: %d", gi->serverRole );
|
|
XP_LOGF( " dictName: %s", gi->dictName );
|
|
XP_LOGF( " isoCode: %s", gi->isoCodeStr );
|
|
}
|
|
}
|
|
#endif
|
|
|
|
#ifdef CPLUS
|
|
}
|
|
#endif
|