Base the number of bytes used to store tiles on the number in the

current dictionary.  This allows us to continue to open games saved
with older code using older dictionaries while still supporting the
new format for up to 64 tiles.  Old versions may crash when opening
games created by new versions, but that's probably ok.
This commit is contained in:
ehouse 2006-08-10 01:21:31 +00:00
parent c41b211735
commit 0b868f5b53
3 changed files with 36 additions and 10 deletions

View file

@ -81,13 +81,14 @@ model_make( MPFORMAL DictionaryCtxt* dict, XW_UtilCtxt* util, XP_U16 nCols,
XP_MEMSET( result, 0, sizeof(*result) ); XP_MEMSET( result, 0, sizeof(*result) );
MPASSIGN(result->vol.mpool, mpool); MPASSIGN(result->vol.mpool, mpool);
result->vol.dict = dict;
result->vol.util = util; result->vol.util = util;
model_init( result, nCols, nRows ); model_init( result, nCols, nRows );
XP_ASSERT( !!util->gameInfo ); XP_ASSERT( !!util->gameInfo );
result->vol.gi = util->gameInfo; result->vol.gi = util->gameInfo;
model_setDictionary( result, dict );
} }
return result; return result;
@ -262,6 +263,12 @@ void
model_setDictionary( ModelCtxt* model, DictionaryCtxt* dict ) model_setDictionary( ModelCtxt* model, DictionaryCtxt* dict )
{ {
model->vol.dict = dict; model->vol.dict = dict;
if ( !!dict ) {
XP_U16 nFaces = dict_numTileFaces( dict );
XP_ASSERT( !!model->vol.stack );
stack_setBitsPerTile( model->vol.stack, nFaces <= 32? 5 : 6 );
}
} /* model_setDictionary */ } /* model_setDictionary */
DictionaryCtxt* DictionaryCtxt*

View file

@ -41,6 +41,7 @@ struct StackCtxt {
XP_U16 cacheNext; XP_U16 cacheNext;
XP_U16 nEntries; XP_U16 nEntries;
XP_U16 bitsPerTile;
XP_U16 highWaterMark; XP_U16 highWaterMark;
MPSLOT MPSLOT
@ -56,6 +57,14 @@ stack_init( StackCtxt* stack )
shrunk to fit as soon as we serialize/deserialize anyway. */ shrunk to fit as soon as we serialize/deserialize anyway. */
} /* stack_init */ } /* stack_init */
void
stack_setBitsPerTile( StackCtxt* stack, XP_U16 bitsPerTile )
{
XP_ASSERT( !!stack );
XP_LOGF( "%s(%d)", __FUNCTION__, bitsPerTile );
stack->bitsPerTile = bitsPerTile;
}
StackCtxt* StackCtxt*
stack_make( MPFORMAL VTableMgr* vtmgr ) stack_make( MPFORMAL VTableMgr* vtmgr )
{ {
@ -118,7 +127,7 @@ stack_writeToStream( StackCtxt* stack, XWStreamCtxt* stream )
static void static void
pushEntry( StackCtxt* stack, const StackEntry* entry ) pushEntry( StackCtxt* stack, const StackEntry* entry )
{ {
XP_U16 i; XP_U16 i, bitsPerTile;
XWStreamPos oldLoc; XWStreamPos oldLoc;
XP_U16 nTiles = entry->u.move.moveInfo.nTiles; XP_U16 nTiles = entry->u.move.moveInfo.nTiles;
XWStreamCtxt* stream = stack->data; XWStreamCtxt* stream = stack->data;
@ -141,11 +150,16 @@ pushEntry( StackCtxt* stack, const StackEntry* entry )
stream_putBits( stream, NTILES_NBITS, nTiles ); stream_putBits( stream, NTILES_NBITS, nTiles );
stream_putBits( stream, 5, entry->u.move.moveInfo.commonCoord ); stream_putBits( stream, 5, entry->u.move.moveInfo.commonCoord );
stream_putBits( stream, 1, entry->u.move.moveInfo.isHorizontal ); stream_putBits( stream, 1, entry->u.move.moveInfo.isHorizontal );
bitsPerTile = stack->bitsPerTile;
XP_ASSERT( bitsPerTile == 5 || bitsPerTile == 6 );
for ( i = 0; i < nTiles; ++i ) { for ( i = 0; i < nTiles; ++i ) {
Tile tile;
stream_putBits( stream, 5, stream_putBits( stream, 5,
entry->u.move.moveInfo.tiles[i].varCoord ); entry->u.move.moveInfo.tiles[i].varCoord );
stream_putBits( stream, TILE_NBITS+1, /* 1 for blank */
entry->u.move.moveInfo.tiles[i].tile ); tile = entry->u.move.moveInfo.tiles[i].tile;
stream_putBits( stream, bitsPerTile, tile & TILE_VALUE_MASK );
stream_putBits( stream, 1, (tile & TILE_BLANK_BIT) != 0 );
} }
if ( entry->moveType == MOVE_TYPE ) { if ( entry->moveType == MOVE_TYPE ) {
traySetToStream( stream, &entry->u.move.newTiles ); traySetToStream( stream, &entry->u.move.newTiles );
@ -174,7 +188,7 @@ pushEntry( StackCtxt* stack, const StackEntry* entry )
static void static void
readEntry( StackCtxt* stack, StackEntry* entry ) readEntry( StackCtxt* stack, StackEntry* entry )
{ {
XP_U16 nTiles, i; XP_U16 nTiles, i, bitsPerTile;
XWStreamCtxt* stream = stack->data; XWStreamCtxt* stream = stack->data;
entry->moveType = (StackMoveType)stream_getBits( stream, 2 ); entry->moveType = (StackMoveType)stream_getBits( stream, 2 );
@ -189,13 +203,17 @@ readEntry( StackCtxt* stack, StackEntry* entry )
XP_ASSERT( nTiles <= MAX_TRAY_TILES ); XP_ASSERT( nTiles <= MAX_TRAY_TILES );
entry->u.move.moveInfo.commonCoord = (XP_U8)stream_getBits(stream, 5); entry->u.move.moveInfo.commonCoord = (XP_U8)stream_getBits(stream, 5);
entry->u.move.moveInfo.isHorizontal = (XP_U8)stream_getBits(stream, 1); entry->u.move.moveInfo.isHorizontal = (XP_U8)stream_getBits(stream, 1);
bitsPerTile = stack->bitsPerTile;
XP_ASSERT( bitsPerTile == 5 || bitsPerTile == 6 );
for ( i = 0; i < nTiles; ++i ) { for ( i = 0; i < nTiles; ++i ) {
Tile tile;
entry->u.move.moveInfo.tiles[i].varCoord = entry->u.move.moveInfo.tiles[i].varCoord =
(XP_U8)stream_getBits(stream, 5); (XP_U8)stream_getBits(stream, 5);
/* PENDING: this is changing from 6 to 7. Need to detect old tile = (Tile)stream_getBits( stream, bitsPerTile );
version and be able to open it!!! */ if ( 0 != stream_getBits( stream, 1 ) ) {
entry->u.move.moveInfo.tiles[i].tile = tile |= TILE_BLANK_BIT;
(Tile)stream_getBits( stream, TILE_NBITS + 1 ); }
entry->u.move.moveInfo.tiles[i].tile = tile;
} }
if ( entry->moveType == MOVE_TYPE ) { if ( entry->moveType == MOVE_TYPE ) {

View file

@ -69,6 +69,7 @@ StackCtxt* stack_make( MPFORMAL VTableMgr* vtmgr );
void stack_destroy( StackCtxt* stack ); void stack_destroy( StackCtxt* stack );
void stack_init( StackCtxt* stack ); void stack_init( StackCtxt* stack );
void stack_setBitsPerTile( StackCtxt* stack, XP_U16 bitsPerTile );
void stack_loadFromStream( StackCtxt* stack, XWStreamCtxt* stream ); void stack_loadFromStream( StackCtxt* stack, XWStreamCtxt* stream );
void stack_writeToStream( StackCtxt* stack, XWStreamCtxt* stream ); void stack_writeToStream( StackCtxt* stack, XWStreamCtxt* stream );