fix bugs around new variable-length tiles array changes, and otherwise

improve storage of col/row data so that test runs succeed with
MAX_COLS of 16 or 32.  Still to do: test mid-game upgrade.
This commit is contained in:
Eric House 2011-11-28 18:46:08 -08:00
parent 42d1d2f7bd
commit adb9b3853d
6 changed files with 146 additions and 118 deletions

View file

@ -207,6 +207,12 @@ board_makeFromStream( MPFORMAL XWStreamCtxt* stream, ModelCtxt* model,
BoardCtxt* board; BoardCtxt* board;
XP_U16 ii; XP_U16 ii;
XP_U16 version = stream_getVersion( stream ); XP_U16 version = stream_getVersion( stream );
XP_U16 nColsNBits;
#ifdef STREAM_VERS_BIGBOARD
nColsNBits = 16 > model_numCols(model) ? NUMCOLS_NBITS_4 : NUMCOLS_NBITS_5;
#else
nColsNBits = NUMCOLS_NBITS_4;
#endif
board = board_make( MPPARM(mpool) model, server, draw, util ); board = board_make( MPPARM(mpool) model, server, draw, util );
@ -238,8 +244,8 @@ board_makeFromStream( MPFORMAL XWStreamCtxt* stream, ModelCtxt* model,
for ( ii = 0; ii < nPlayers; ++ii ) { for ( ii = 0; ii < nPlayers; ++ii ) {
PerTurnInfo* pti = &board->pti[ii]; PerTurnInfo* pti = &board->pti[ii];
BoardArrow* arrow = &pti->boardArrow; BoardArrow* arrow = &pti->boardArrow;
arrow->col = (XP_U8)stream_getBits( stream, NUMCOLS_NBITS ); arrow->col = (XP_U8)stream_getBits( stream, nColsNBits );
arrow->row = (XP_U8)stream_getBits( stream, NUMCOLS_NBITS ); arrow->row = (XP_U8)stream_getBits( stream, nColsNBits );
arrow->vert = (XP_Bool)stream_getBits( stream, 1 ); arrow->vert = (XP_Bool)stream_getBits( stream, 1 );
arrow->visible = (XP_Bool)stream_getBits( stream, 1 ); arrow->visible = (XP_Bool)stream_getBits( stream, 1 );
@ -284,7 +290,15 @@ board_makeFromStream( MPFORMAL XWStreamCtxt* stream, ModelCtxt* model,
void void
board_writeToStream( BoardCtxt* board, XWStreamCtxt* stream ) board_writeToStream( BoardCtxt* board, XWStreamCtxt* stream )
{ {
XP_U16 nPlayers, i; XP_U16 nPlayers, ii;
XP_U16 nColsNBits;
#ifdef STREAM_VERS_BIGBOARD
nColsNBits = 16 > model_numCols(board->model) ? NUMCOLS_NBITS_4
: NUMCOLS_NBITS_5;
#else
nColsNBits = NUMCOLS_NBITS_4;
#endif
stream_putBits( stream, 4, board->sd[SCROLL_H].offset ); stream_putBits( stream, 4, board->sd[SCROLL_H].offset );
stream_putBits( stream, 4, board->zoomCount ); stream_putBits( stream, 4, board->zoomCount );
@ -304,11 +318,11 @@ board_writeToStream( BoardCtxt* board, XWStreamCtxt* stream )
XP_ASSERT( !!board->server ); XP_ASSERT( !!board->server );
nPlayers = board->gi->nPlayers; nPlayers = board->gi->nPlayers;
for ( i = 0; i < nPlayers; ++i ) { for ( ii = 0; ii < nPlayers; ++ii ) {
PerTurnInfo* pti = &board->pti[i]; PerTurnInfo* pti = &board->pti[ii];
BoardArrow* arrow = &pti->boardArrow; BoardArrow* arrow = &pti->boardArrow;
stream_putBits( stream, NUMCOLS_NBITS, arrow->col ); stream_putBits( stream, nColsNBits, arrow->col );
stream_putBits( stream, NUMCOLS_NBITS, arrow->row ); stream_putBits( stream, nColsNBits, arrow->row );
stream_putBits( stream, 1, arrow->vert ); stream_putBits( stream, 1, arrow->vert );
stream_putBits( stream, 1, arrow->visible ); stream_putBits( stream, 1, arrow->visible );

View file

@ -1,6 +1,6 @@
/* -*- compile-command: "cd ../linux && make MEMDEBUG=TRUE"; -*- */ /* -*- compile-command: "cd ../linux && make -j3 MEMDEBUG=TRUE"; -*- */
/* /*
* Copyright 2001-2009 by Eric House (xwords@eehouse.org). All rights * Copyright 2001-2011 by Eric House (xwords@eehouse.org). All rights
* reserved. * reserved.
* *
* This program is free software; you can redistribute it and/or * This program is free software; you can redistribute it and/or
@ -90,7 +90,7 @@ game_makeNewGame( MPFORMAL XWGame* game, CurGameInfo* gi,
gi->gameID = gameID; gi->gameID = gameID;
game->model = model_make( MPPARM(mpool) (DictionaryCtxt*)NULL, NULL, util, game->model = model_make( MPPARM(mpool) (DictionaryCtxt*)NULL, NULL, util,
gi->boardSize, gi->boardSize ); gi->boardSize );
#ifndef XWFEATURE_STANDALONE_ONLY #ifndef XWFEATURE_STANDALONE_ONLY
if ( gi->serverRole != SERVER_STANDALONE ) { if ( gi->serverRole != SERVER_STANDALONE ) {
@ -162,7 +162,7 @@ game_reset( MPFORMAL XWGame* game, CurGameInfo* gi,
# endif # endif
#endif #endif
model_init( game->model, gi->boardSize, gi->boardSize ); model_setSize( game->model, gi->boardSize );
server_reset( game->server, server_reset( game->server,
#ifndef XWFEATURE_STANDALONE_ONLY #ifndef XWFEATURE_STANDALONE_ONLY
game->comms game->comms
@ -259,6 +259,7 @@ game_saveToStream( const XWGame* game, const CurGameInfo* gi,
XWStreamCtxt* stream ) XWStreamCtxt* stream )
{ {
stream_putU8( stream, CUR_STREAM_VERS ); stream_putU8( stream, CUR_STREAM_VERS );
stream_setVersion( stream, CUR_STREAM_VERS );
gi_writeToStream( stream, gi ); gi_writeToStream( stream, gi );
@ -426,10 +427,13 @@ gi_readFromStream( MPFORMAL XWStreamCtxt* stream, CurGameInfo* gi )
XP_U16 ii; XP_U16 ii;
XP_UCHAR* str; XP_UCHAR* str;
XP_U16 strVersion = stream_getVersion( stream ); XP_U16 strVersion = stream_getVersion( stream );
XP_U16 nColsNBits;
XP_ASSERT( 0 < strVersion );
#ifdef STREAM_VERS_BIGBOARD #ifdef STREAM_VERS_BIGBOARD
XP_U16 nColsNBits = STREAM_VERS_BIGBOARD > strVersion ? 4 : NUMCOLS_NBITS; nColsNBits = STREAM_VERS_BIGBOARD > strVersion ? NUMCOLS_NBITS_4
: NUMCOLS_NBITS_5;
#else #else
XP_U16 nColsNBits = NUMCOLS_NBITS; nColsNBits = NUMCOLS_NBITS_4;
#endif #endif
str = stringFromStream( mpool, stream ); str = stringFromStream( mpool, stream );
@ -495,16 +499,16 @@ gi_writeToStream( XWStreamCtxt* stream, const CurGameInfo* gi )
{ {
const LocalPlayer* pl; const LocalPlayer* pl;
XP_U16 ii; XP_U16 ii;
XP_U16 nColsNBits;
#ifdef STREAM_VERS_BIGBOARD #ifdef STREAM_VERS_BIGBOARD
XP_U16 strVersion = stream_getVersion( stream ); XP_U16 strVersion = stream_getVersion( stream );
XP_ASSERT( STREAM_SAVE_PREVWORDS <= strVersion ); XP_ASSERT( STREAM_SAVE_PREVWORDS <= strVersion );
XP_U16 nColsNBits = STREAM_VERS_BIGBOARD > strVersion ? 4 : NUMCOLS_NBITS; nColsNBits = STREAM_VERS_BIGBOARD > strVersion ? NUMCOLS_NBITS_4
: NUMCOLS_NBITS_5;
#else #else
XP_U16 nColsNBits = NUMCOLS_NBITS; nColsNBits = NUMCOLS_NBITS_4;
#endif #endif
stringToStream( stream, gi->dictName ); stringToStream( stream, gi->dictName );
stream_putBits( stream, NPLAYERS_NBITS, gi->nPlayers ); stream_putBits( stream, NPLAYERS_NBITS, gi->nPlayers );

View file

@ -71,9 +71,10 @@ static void buildModelFromStack( ModelCtxt* model, StackCtxt* stack,
MovePrintFuncPre mpfpr, MovePrintFuncPre mpfpr,
MovePrintFuncPost mpfpo, void* closure ); MovePrintFuncPost mpfpo, void* closure );
static void setPendingCounts( ModelCtxt* model, XP_S16 turn ); static void setPendingCounts( ModelCtxt* model, XP_S16 turn );
static void loadPlayerCtxt( XWStreamCtxt* stream, XP_U16 version, static void loadPlayerCtxt( const ModelCtxt* model, XWStreamCtxt* stream,
XP_U16 version, PlayerCtxt* pc );
static void writePlayerCtxt( const ModelCtxt* model, XWStreamCtxt* stream,
PlayerCtxt* pc ); PlayerCtxt* pc );
static void writePlayerCtxt( XWStreamCtxt* stream, PlayerCtxt* pc );
static XP_U16 model_getRecentPassCount( ModelCtxt* model ); static XP_U16 model_getRecentPassCount( ModelCtxt* model );
static XP_Bool recordWord( const XP_UCHAR* word, XP_Bool isLegal, static XP_Bool recordWord( const XP_UCHAR* word, XP_Bool isLegal,
#ifdef XWFEATURE_BOARDWORDS #ifdef XWFEATURE_BOARDWORDS
@ -85,13 +86,10 @@ static XP_Bool recordWord( const XP_UCHAR* word, XP_Bool isLegal,
* *
****************************************************************************/ ****************************************************************************/
ModelCtxt* ModelCtxt*
model_make( MPFORMAL DictionaryCtxt* dict, model_make( MPFORMAL DictionaryCtxt* dict, const PlayerDicts* dicts,
const PlayerDicts* dicts, XW_UtilCtxt* util, XP_U16 nCols, XW_UtilCtxt* util, XP_U16 nCols )
XP_U16 nRows )
{ {
ModelCtxt* result; ModelCtxt* result = (ModelCtxt*)XP_MALLOC( mpool, sizeof( *result ) );
XP_U16 size = sizeof(*result) + TILES_SIZE(result, nCols);
result = (ModelCtxt*)XP_MALLOC( mpool, size );
if ( result != NULL ) { if ( result != NULL ) {
XP_MEMSET( result, 0, sizeof(*result) ); XP_MEMSET( result, 0, sizeof(*result) );
MPASSIGN(result->vol.mpool, mpool); MPASSIGN(result->vol.mpool, mpool);
@ -100,7 +98,7 @@ model_make( MPFORMAL DictionaryCtxt* dict,
result->vol.wni.proc = recordWord; result->vol.wni.proc = recordWord;
result->vol.wni.closure = &result->vol.rwi; result->vol.wni.closure = &result->vol.rwi;
model_init( result, nCols, nRows ); model_setSize( result, nCols );
XP_ASSERT( !!util->gameInfo ); XP_ASSERT( !!util->gameInfo );
result->vol.gi = util->gameInfo; result->vol.gi = util->gameInfo;
@ -121,18 +119,21 @@ model_makeFromStream( MPFORMAL XWStreamCtxt* stream, DictionaryCtxt* dict,
XP_U16 ii; XP_U16 ii;
XP_Bool hasDict; XP_Bool hasDict;
XP_U16 nPlayers; XP_U16 nPlayers;
XP_U16 nColsNBits;
XP_U16 version = stream_getVersion( stream ); XP_U16 version = stream_getVersion( stream );
#ifdef STREAM_VERS_BIGBOARD
nColsNBits = STREAM_VERS_BIGBOARD > version ? 4 : NUMCOLS_NBITS;
#else
nColsNBits = NUMCOLS_NBITS;
#endif
XP_ASSERT( !!dict || !!dicts ); XP_ASSERT( !!dict || !!dicts );
nCols = (XP_U16)stream_getBits( stream, nColsNBits ); if ( 0 ) {
nRows = (XP_U16)stream_getBits( stream, nColsNBits ); #ifdef STREAM_VERS_BIGBOARD
} else if ( STREAM_VERS_BIGBOARD <= version ) {
nCols = (XP_U16)stream_getBits( stream, NUMCOLS_NBITS_5 );
nRows = nCols;
#endif
} else {
nCols = (XP_U16)stream_getBits( stream, NUMCOLS_NBITS_4 );
nRows = (XP_U16)stream_getBits( stream, NUMCOLS_NBITS_4 );
}
XP_ASSERT( MAX_COLS >= nCols );
hasDict = (version >= STREAM_VERS_MODEL_NO_DICT) hasDict = (version >= STREAM_VERS_MODEL_NO_DICT)
? XP_FALSE : stream_getBits( stream, 1 ); ? XP_FALSE : stream_getBits( stream, 1 );
@ -144,18 +145,18 @@ model_makeFromStream( MPFORMAL XWStreamCtxt* stream, DictionaryCtxt* dict,
dict_destroy( savedDict ); dict_destroy( savedDict );
} }
model = model_make( MPPARM(mpool) dict, dicts, util, nCols, nRows ); model = model_make( MPPARM(mpool) dict, dicts, util, nCols );
model->nPlayers = nPlayers; model->nPlayers = nPlayers;
#ifdef STREAM_VERS_BIGBOARD #ifdef STREAM_VERS_BIGBOARD
if ( STREAM_VERS_BIGBOARD <= version ) { if ( STREAM_VERS_BIGBOARD <= version ) {
model->nBonuses = stream_getBits( stream, 7 ); model->vol.nBonuses = stream_getBits( stream, 7 );
if ( 0 < model->nBonuses ) { if ( 0 < model->vol.nBonuses ) {
model->bonuses = model->vol.bonuses =
XP_MALLOC( model->vol.mpool, XP_MALLOC( model->vol.mpool,
model->nBonuses * sizeof( model->bonuses[0] ) ); model->vol.nBonuses * sizeof( model->vol.bonuses[0] ) );
for ( ii = 0; ii < model->nBonuses; ++ii ) { for ( ii = 0; ii < model->vol.nBonuses; ++ii ) {
model->bonuses[ii] = stream_getBits( stream, 4 ); model->vol.bonuses[ii] = stream_getBits( stream, 4 );
} }
} }
} }
@ -169,7 +170,7 @@ model_makeFromStream( MPFORMAL XWStreamCtxt* stream, DictionaryCtxt* dict,
NULL ); NULL );
for ( ii = 0; ii < model->nPlayers; ++ii ) { for ( ii = 0; ii < model->nPlayers; ++ii ) {
loadPlayerCtxt( stream, version, &model->players[ii] ); loadPlayerCtxt( model, stream, version, &model->players[ii] );
setPendingCounts( model, ii ); setPendingCounts( model, ii );
invalidateScore( model, ii ); invalidateScore( model, ii );
} }
@ -181,24 +182,28 @@ void
model_writeToStream( ModelCtxt* model, XWStreamCtxt* stream ) model_writeToStream( ModelCtxt* model, XWStreamCtxt* stream )
{ {
XP_U16 ii; XP_U16 ii;
#ifdef STREAM_VERS_BIGBOARD
stream_putBits( stream, NUMCOLS_NBITS, model->nCols ); XP_ASSERT( STREAM_VERS_BIGBOARD <= stream_getVersion( stream ) );
stream_putBits( stream, NUMCOLS_NBITS, model->nRows ); stream_putBits( stream, NUMCOLS_NBITS_5, model->nCols );
#else
stream_putBits( stream, NUMCOLS_NBITS_4, model->nCols );
stream_putBits( stream, NUMCOLS_NBITS_4, model->nRows );
#endif
/* we have two bits for nPlayers, so range must be 0..3, not 1..4 */ /* we have two bits for nPlayers, so range must be 0..3, not 1..4 */
stream_putBits( stream, NPLAYERS_NBITS, model->nPlayers ); stream_putBits( stream, NPLAYERS_NBITS, model->nPlayers );
#ifdef STREAM_VERS_BIGBOARD #ifdef STREAM_VERS_BIGBOARD
stream_putBits( stream, 7, model->nBonuses ); stream_putBits( stream, 7, model->vol.nBonuses );
for ( ii = 0; ii < model->nBonuses; ++ii ) { for ( ii = 0; ii < model->vol.nBonuses; ++ii ) {
stream_putBits( stream, 4, model->bonuses[ii] ); stream_putBits( stream, 4, model->vol.bonuses[ii] );
} }
#endif #endif
stack_writeToStream( model->vol.stack, stream ); stack_writeToStream( model->vol.stack, stream );
for ( ii = 0; ii < model->nPlayers; ++ii ) { for ( ii = 0; ii < model->nPlayers; ++ii ) {
writePlayerCtxt( stream, &model->players[ii] ); writePlayerCtxt( model, stream, &model->players[ii] );
} }
} /* model_writeToStream */ } /* model_writeToStream */
@ -244,35 +249,44 @@ model_writeToTextStream( const ModelCtxt* model, XWStreamCtxt* stream )
#endif #endif
void void
model_init( ModelCtxt* model, XP_U16 nCols, XP_U16 nRows ) model_setSize( ModelCtxt* model, XP_U16 nCols )
{ {
ModelVolatiles vol = model->vol; /* save vol so we don't wipe it out */ ModelVolatiles vol = model->vol; /* save vol so we don't wipe it out */
XP_U16 oldSize = model->nCols; /* zero when called from model_make() */
XP_ASSERT( MAX_COLS >= nCols );
XP_ASSERT( model != NULL ); XP_ASSERT( model != NULL );
XP_MEMSET( model, 0, sizeof( *model ) ); XP_MEMSET( model, 0, sizeof( *model ) );
XP_MEMSET( &model->tiles, TILE_EMPTY_BIT, TILES_SIZE(model, nCols) );
model->nCols = nCols; model->nCols = nCols;
model->nRows = nRows; model->nRows = nCols;
model->vol = vol; model->vol = vol;
if ( oldSize != nCols ) {
if ( !!model->vol.tiles ) {
XP_FREE( model->vol.mpool, model->vol.tiles );
}
model->vol.tiles = XP_MALLOC( model->vol.mpool, TILES_SIZE(model, nCols) );
}
XP_MEMSET( model->vol.tiles, TILE_EMPTY_BIT, TILES_SIZE(model, nCols) );
if ( !!model->vol.stack ) { if ( !!model->vol.stack ) {
stack_init( model->vol.stack ); stack_init( model->vol.stack );
} else { } else {
model->vol.stack = stack_make( MPPARM(model->vol.mpool) model->vol.stack = stack_make( MPPARM(model->vol.mpool)
util_getVTManager(model->vol.util)); util_getVTManager(model->vol.util));
} }
} /* model_init */ } /* model_setSize */
void void
model_destroy( ModelCtxt* model ) model_destroy( ModelCtxt* model )
{ {
stack_destroy( model->vol.stack ); stack_destroy( model->vol.stack );
/* is this it!? */ /* is this it!? */
if ( !!model->bonuses ) { if ( !!model->vol.bonuses ) {
XP_FREE( model->vol.mpool, model->bonuses ); XP_FREE( model->vol.mpool, model->vol.bonuses );
} }
XP_FREE( model->vol.mpool, model->vol.tiles );
XP_FREE( model->vol.mpool, model ); XP_FREE( model->vol.mpool, model );
} /* model_destroy */ } /* model_destroy */
@ -290,42 +304,28 @@ model_setSquareBonuses( ModelCtxt* model, XWBonusType* bonuses, XP_U16 nBonuses
XP_ASSERT( wantLen == nBonuses ); XP_ASSERT( wantLen == nBonuses );
#endif #endif
if ( !!model->bonuses ) { if ( !!model->vol.bonuses ) {
XP_FREE( model->vol.mpool, model->bonuses ); XP_FREE( model->vol.mpool, model->vol.bonuses );
} }
model->bonuses = XP_MALLOC( model->vol.mpool, model->vol.bonuses = XP_MALLOC( model->vol.mpool,
nBonuses * sizeof(model->bonuses[0]) ); nBonuses * sizeof(model->vol.bonuses[0]) );
XP_MEMCPY( model->bonuses, bonuses, nBonuses * sizeof(model->bonuses[0]) ); XP_MEMCPY( model->vol.bonuses, bonuses,
model->nBonuses = nBonuses; nBonuses * sizeof(model->vol.bonuses[0]) );
model->vol.nBonuses = nBonuses;
} }
static void
borrowSquareBonuses( ModelCtxt* dest, const ModelCtxt* src )
{
XP_ASSERT( !dest->bonuses );
dest->bonuses = src->bonuses;
dest->nBonuses = src->nBonuses;
}
static void
returnSquareBonuses( ModelCtxt* dest )
{
dest->bonuses = NULL;
dest->nBonuses = 0;
}
#else
# define borrowSquareBonuses(d,s)
# define returnSquareBonuses(d)
#endif #endif
XWBonusType XWBonusType
model_getSquareBonus( const ModelCtxt* model, XP_U16 col, XP_U16 row ) model_getSquareBonus( const ModelCtxt* model, XP_U16 col, XP_U16 row )
{ {
XWBonusType result = BONUS_NONE; XWBonusType result = BONUS_NONE;
#ifdef STREAM_VERS_BIGBOARD
const ModelCtxt* bonusOwner = model->loaner? model->loaner : model;
#endif
if ( 0 ) { if ( 0 ) {
#ifdef STREAM_VERS_BIGBOARD #ifdef STREAM_VERS_BIGBOARD
} else if ( !!model->bonuses ) { } else if ( !!bonusOwner->vol.bonuses ) {
XP_U16 nCols = model_numCols( model ); XP_U16 nCols = model_numCols( model );
XP_U16 ii; XP_U16 ii;
if ( col > (nCols/2) ) { if ( col > (nCols/2) ) {
@ -343,8 +343,8 @@ model_getSquareBonus( const ModelCtxt* model, XP_U16 col, XP_U16 row )
col += ii; col += ii;
} }
if ( col < model->nBonuses ) { if ( col < bonusOwner->vol.nBonuses ) {
result = model->bonuses[col]; result = bonusOwner->vol.bonuses[col];
} }
#endif #endif
} else { } else {
@ -694,7 +694,7 @@ setModelTileRaw( ModelCtxt* model, XP_U16 col, XP_U16 row, CellTile tile )
{ {
XP_ASSERT( col < model->nCols ); XP_ASSERT( col < model->nCols );
XP_ASSERT( row < model->nRows ); XP_ASSERT( row < model->nRows );
model->tiles[(row*model->nCols) + col] = tile; model->vol.tiles[(row*model->nCols) + col] = tile;
} /* model_setTile */ } /* model_setTile */
static CellTile static CellTile
@ -704,7 +704,7 @@ getModelTileRaw( const ModelCtxt* model, XP_U16 col, XP_U16 row )
XP_U16 nCols = model->nCols; XP_U16 nCols = model->nCols;
XP_ASSERT( model->nRows == nCols ); XP_ASSERT( model->nRows == nCols );
if ( col < nCols && row < nCols ) { if ( col < nCols && row < nCols ) {
tile = model->tiles[(row*nCols) + col]; tile = model->vol.tiles[(row*nCols) + col];
} else { } else {
tile = TILE_EMPTY_BIT; tile = TILE_EMPTY_BIT;
} }
@ -932,11 +932,12 @@ model_currentMoveToStream( ModelCtxt* model, XP_S16 turn,
{ {
PlayerCtxt* player; PlayerCtxt* player;
XP_S16 numTiles; XP_S16 numTiles;
XP_U16 nColsNBits;
#ifdef STREAM_VERS_BIGBOARD #ifdef STREAM_VERS_BIGBOARD
XP_U16 version = stream_getVersion( stream ); nColsNBits = 16 <= model_numCols( model ) ? NUMCOLS_NBITS_5
XP_U16 nColsNBits = STREAM_VERS_BIGBOARD > version ? 4 : NUMCOLS_NBITS; : NUMCOLS_NBITS_4;
#else #else
XP_U16 nColsNBits = NUMCOLS_NBITS; nColsNBits = NUMCOLS_NBITS_4;
#endif #endif
XP_ASSERT( turn >= 0 ); XP_ASSERT( turn >= 0 );
@ -972,14 +973,14 @@ model_makeTurnFromStream( ModelCtxt* model, XP_U16 playerNum,
{ {
XP_U16 numTiles, ii; XP_U16 numTiles, ii;
Tile blank = dict_getBlankTile( model_getDictionary(model) ); Tile blank = dict_getBlankTile( model_getDictionary(model) );
XP_U16 nColsNBits;
#ifdef STREAM_VERS_BIGBOARD #ifdef STREAM_VERS_BIGBOARD
XP_U16 version = stream_getVersion( stream ); nColsNBits = 16 <= model_numCols( model ) ? NUMCOLS_NBITS_5
XP_U16 nColsNBits = STREAM_VERS_BIGBOARD > version ? 4 : NUMCOLS_NBITS; : NUMCOLS_NBITS_4;
#else #else
XP_U16 nColsNBits = NUMCOLS_NBITS; nColsNBits = NUMCOLS_NBITS_4;
#endif #endif
model_resetCurrentTurn( model, playerNum ); model_resetCurrentTurn( model, playerNum );
numTiles = (XP_U16)stream_getBits( stream, NTILES_NBITS ); numTiles = (XP_U16)stream_getBits( stream, NTILES_NBITS );
@ -2058,10 +2059,9 @@ makeTmpModel( ModelCtxt* model, XWStreamCtxt* stream,
{ {
ModelCtxt* tmpModel = model_make( MPPARM(model->vol.mpool) ModelCtxt* tmpModel = model_make( MPPARM(model->vol.mpool)
model_getDictionary(model), NULL, model_getDictionary(model), NULL,
model->vol.util, model_numCols(model), model->vol.util, model_numCols(model) );
model_numRows(model)); tmpModel->loaner = model;
model_setNPlayers( tmpModel, model->nPlayers ); model_setNPlayers( tmpModel, model->nPlayers );
borrowSquareBonuses( tmpModel, model );
buildModelFromStack( tmpModel, model->vol.stack, XP_FALSE, 0, stream, buildModelFromStack( tmpModel, model->vol.stack, XP_FALSE, 0, stream,
(WordNotifierInfo*)NULL, mpf_pre, mpf_post, closure ); (WordNotifierInfo*)NULL, mpf_pre, mpf_post, closure );
@ -2083,7 +2083,6 @@ model_writeGameHistory( ModelCtxt* model, XWStreamCtxt* stream,
tmpModel = makeTmpModel( model, stream, printMovePre, printMovePost, tmpModel = makeTmpModel( model, stream, printMovePre, printMovePost,
&closure ); &closure );
returnSquareBonuses( tmpModel );
model_destroy( tmpModel ); model_destroy( tmpModel );
if ( gameOver ) { if ( gameOver ) {
@ -2343,14 +2342,18 @@ model_getPlayersLastScore( ModelCtxt* model, XP_S16 player,
} /* model_getPlayersLastScore */ } /* model_getPlayersLastScore */
static void static void
loadPlayerCtxt( XWStreamCtxt* stream, XP_U16 version, PlayerCtxt* pc ) loadPlayerCtxt( const ModelCtxt* model, XWStreamCtxt* stream, XP_U16 version,
PlayerCtxt* pc )
{ {
PendingTile* pt; PendingTile* pt;
XP_U16 nTiles; XP_U16 nTiles;
XP_U16 nColsNBits;
#ifdef STREAM_VERS_BIGBOARD #ifdef STREAM_VERS_BIGBOARD
XP_U16 nColsNBits = STREAM_VERS_BIGBOARD > version ? 4 : NUMCOLS_NBITS; nColsNBits = 16 <= model_numCols( model ) ? NUMCOLS_NBITS_5
: NUMCOLS_NBITS_4;
#else #else
XP_U16 nColsNBits = NUMCOLS_NBITS; XP_USE(model);
nColsNBits = NUMCOLS_NBITS_4;
#endif #endif
pc->curMoveValid = stream_getBits( stream, 1 ); pc->curMoveValid = stream_getBits( stream, 1 );
@ -2377,10 +2380,18 @@ loadPlayerCtxt( XWStreamCtxt* stream, XP_U16 version, PlayerCtxt* pc )
} /* loadPlayerCtxt */ } /* loadPlayerCtxt */
static void static void
writePlayerCtxt( XWStreamCtxt* stream, PlayerCtxt* pc ) writePlayerCtxt( const ModelCtxt* model, XWStreamCtxt* stream, PlayerCtxt* pc )
{ {
XP_U16 nTiles; XP_U16 nTiles;
PendingTile* pt; PendingTile* pt;
XP_U16 nColsNBits;
#ifdef STREAM_VERS_BIGBOARD
nColsNBits = 16 <= model_numCols( model ) ? NUMCOLS_NBITS_5
: NUMCOLS_NBITS_4;
#else
XP_USE(model);
nColsNBits = NUMCOLS_NBITS_4;
#endif
stream_putBits( stream, 1, pc->curMoveValid ); stream_putBits( stream, 1, pc->curMoveValid );
@ -2391,8 +2402,8 @@ writePlayerCtxt( XWStreamCtxt* stream, PlayerCtxt* pc )
nTiles = pc->nPending + pc->nUndone; nTiles = pc->nPending + pc->nUndone;
for ( pt = pc->pendingTiles; nTiles-- > 0; ++pt ) { for ( pt = pc->pendingTiles; nTiles-- > 0; ++pt ) {
stream_putBits( stream, NUMCOLS_NBITS, pt->col ); stream_putBits( stream, nColsNBits, pt->col );
stream_putBits( stream, NUMCOLS_NBITS, pt->row ); stream_putBits( stream, nColsNBits, pt->row );
stream_putBits( stream, 7, pt->tile ); stream_putBits( stream, 7, pt->tile );
} }
} /* writePlayerCtxt */ } /* writePlayerCtxt */

View file

@ -29,10 +29,9 @@
extern "C" { extern "C" {
#endif #endif
#if MAX_COLS <= 16 #define NUMCOLS_NBITS_4 4
# define NUMCOLS_NBITS 4 #if 16 < MAX_COLS && MAX_COLS <= 32
#elif MAX_COLS <= 32 # define NUMCOLS_NBITS_5 5
# define NUMCOLS_NBITS 5
#endif #endif
#ifdef EIGHT_TILES #ifdef EIGHT_TILES
@ -101,7 +100,7 @@ typedef XP_U8 TileBit; /* bits indicating selection of tiles in tray */
ModelCtxt* model_make( MPFORMAL DictionaryCtxt* dict, const PlayerDicts* dicts, ModelCtxt* model_make( MPFORMAL DictionaryCtxt* dict, const PlayerDicts* dicts,
XW_UtilCtxt* util, XP_U16 nCols, XP_U16 nRows ); XW_UtilCtxt* util, XP_U16 nCols );
ModelCtxt* model_makeFromStream( MPFORMAL XWStreamCtxt* stream, ModelCtxt* model_makeFromStream( MPFORMAL XWStreamCtxt* stream,
DictionaryCtxt* dict, const PlayerDicts* dicts, DictionaryCtxt* dict, const PlayerDicts* dicts,
@ -113,7 +112,7 @@ void model_writeToStream( ModelCtxt* model, XWStreamCtxt* stream );
void model_writeToTextStream( const ModelCtxt* model, XWStreamCtxt* stream ); void model_writeToTextStream( const ModelCtxt* model, XWStreamCtxt* stream );
#endif #endif
void model_init( ModelCtxt* model, XP_U16 nCols, XP_U16 nRows ); void model_setSize( ModelCtxt* model, XP_U16 boardSize );
void model_destroy( ModelCtxt* model ); void model_destroy( ModelCtxt* model );
void model_setNPlayers( ModelCtxt* model, XP_U16 numPlayers ); void model_setNPlayers( ModelCtxt* model, XP_U16 numPlayers );
XP_U16 model_getNPlayers( const ModelCtxt* model ); XP_U16 model_getNPlayers( const ModelCtxt* model );

View file

@ -64,6 +64,11 @@ typedef struct ModelVolatiles {
RecordWordsInfo rwi; RecordWordsInfo rwi;
WordNotifierInfo wni; WordNotifierInfo wni;
XP_U16 nTilesOnBoard; XP_U16 nTilesOnBoard;
CellTile* tiles;
XP_U16 nBonuses;
XWBonusType* bonuses;
MPSLOT MPSLOT
} ModelVolatiles; } ModelVolatiles;
@ -75,13 +80,11 @@ struct ModelCtxt {
XP_U16 nPlayers; XP_U16 nPlayers;
XP_U16 nCols; XP_U16 nCols;
XP_U16 nRows; XP_U16 nRows;
XP_U16 nBonuses;
XWBonusType* bonuses;
CellTile tiles[]; const ModelCtxt* loaner; /* allows sharing bonuses */
}; };
#define TILES_SIZE(m,nc) ((nc) * (nc) * sizeof((m)->tiles[0])) #define TILES_SIZE(m,nc) ((nc) * (nc) * sizeof((m)->vol.tiles[0]))
void invalidateScore( ModelCtxt* model, XP_S16 player ); void invalidateScore( ModelCtxt* model, XP_S16 player );
XP_Bool tilesInLine( ModelCtxt* model, XP_S16 turn, XP_Bool* isHorizontal ); XP_Bool tilesInLine( ModelCtxt* model, XP_S16 turn, XP_Bool* isHorizontal );

View file

@ -1199,8 +1199,7 @@ client_readInitialMessage( ServerCtxt* server, XWStreamCtxt* stream )
XP_ASSERT( channelNo != 0 ); XP_ASSERT( channelNo != 0 );
server->nv.addresses[0].channelNo = channelNo; server->nv.addresses[0].channelNo = channelNo;
/* PENDING init's a bit harsh for setting the size */ model_setSize( model, nCols );
model_init( model, nCols, nCols );
nPlayers = localGI.nPlayers; nPlayers = localGI.nPlayers;
XP_STATUSF( "reading in %d players", localGI.nPlayers ); XP_STATUSF( "reading in %d players", localGI.nPlayers );
@ -1418,9 +1417,6 @@ messageStreamWithHeader( ServerCtxt* server, XP_U16 devIndex, XW_Proto code )
printCode("making", code); printCode("making", code);
stream = util_makeStreamFromAddr( server->vol.util, channelNo ); stream = util_makeStreamFromAddr( server->vol.util, channelNo );
#ifdef STREAM_VERS_BIGBOARD
stream_setVersion( stream, server->nv.streamVersion );
#endif
stream_open( stream ); stream_open( stream );
writeProto( server, stream, code ); writeProto( server, stream, code );
@ -2528,6 +2524,7 @@ writeProto( const ServerCtxt* server, XWStreamCtxt* stream, XW_Proto proto )
stream_putBits( stream, XWPROTO_NBITS, XWPROTO_NEW_PROTO ); stream_putBits( stream, XWPROTO_NBITS, XWPROTO_NEW_PROTO );
stream_putBits( stream, 8, CUR_STREAM_VERS ); stream_putBits( stream, 8, CUR_STREAM_VERS );
} }
stream_setVersion( stream, server->nv.streamVersion );
#else #else
XP_USE(server); XP_USE(server);
#endif #endif