mirror of
git://xwords.git.sourceforge.net/gitroot/xwords/xwords
synced 2025-01-10 05:26:10 +01:00
1346aa6ee5
assigned. Later '3' will be replaced by the location of the tray divider, but that has to be moved from board to model first.
506 lines
14 KiB
C
506 lines
14 KiB
C
/* -*- compile-command: "cd ../linux && make -j3 MEMDEBUG=TRUE"; -*- */
|
|
/*
|
|
* Copyright 2001, 2006-2012 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 "modelp.h"
|
|
|
|
#include "mempool.h"
|
|
#include "xwstream.h"
|
|
#include "movestak.h"
|
|
#include "memstream.h"
|
|
#include "strutils.h"
|
|
#include "dbgutil.h"
|
|
|
|
/* HASH_STREAM: It should be possible to hash the move stack by simply hashing
|
|
the stream from the beginning to the top of the undo stack (excluding
|
|
what's in the redo area), avoiding iterating over it and doing a ton of
|
|
bitwise operations to read it into entries. But I don't currently seem to
|
|
have a XWStreamPos that corresponds to the undo-top and so I can't figure
|
|
out the length. Hashing that includes the redo part of the stack doesn't
|
|
work once there's been undo activity. (Not sure why...) */
|
|
|
|
#ifdef CPLUS
|
|
extern "C" {
|
|
#endif
|
|
|
|
struct StackCtxt {
|
|
VTableMgr* vtmgr;
|
|
|
|
XWStreamCtxt* data;
|
|
|
|
XWStreamPos top;
|
|
|
|
XWStreamPos cachedPos;
|
|
|
|
XP_U16 cacheNext;
|
|
XP_U16 nEntries;
|
|
XP_U16 bitsPerTile;
|
|
XP_U16 highWaterMark;
|
|
|
|
DIRTY_SLOT
|
|
MPSLOT
|
|
};
|
|
|
|
void
|
|
stack_init( StackCtxt* stack )
|
|
{
|
|
stack->nEntries = stack->highWaterMark = 0;
|
|
stack->top = START_OF_STREAM;
|
|
|
|
/* I see little point in freeing or shrinking stack->data. It'll get
|
|
shrunk to fit as soon as we serialize/deserialize anyway. */
|
|
} /* stack_init */
|
|
|
|
static XP_U32
|
|
augmentHash( XP_U32 hash, const XP_U8* ptr, XP_U16 len )
|
|
{
|
|
XP_ASSERT( 0 < len );
|
|
// see http://en.wikipedia.org/wiki/Jenkins_hash_function
|
|
XP_U16 ii;
|
|
for ( ii = 0; ii < len; ++ii ) {
|
|
hash += *ptr++;
|
|
hash += (hash << 10);
|
|
hash ^= (hash >> 6);
|
|
}
|
|
// XP_LOGF( "%s: hashed %d bytes -> %X", __func__, len, (unsigned int)hash );
|
|
return hash;
|
|
}
|
|
|
|
static XP_U32
|
|
finishHash( XP_U32 hash )
|
|
{
|
|
hash += (hash << 3);
|
|
hash ^= (hash >> 11);
|
|
hash += (hash << 15);
|
|
return hash;
|
|
}
|
|
|
|
static XP_U32
|
|
augmentFor( XP_U32 hash, const StackEntry* entry )
|
|
{
|
|
switch( entry->moveType ) {
|
|
case ASSIGN_TYPE: {
|
|
TrayTileSet tiles;
|
|
sortTiles( &tiles, &entry->u.assign.tiles, 0 );
|
|
hash = augmentHash( hash, (XP_U8*)&tiles, sizeof(tiles) );
|
|
}
|
|
break;
|
|
case MOVE_TYPE:
|
|
hash = augmentHash( hash, (XP_U8*)&entry->u.move,
|
|
sizeof(entry->u.move) );
|
|
break;
|
|
case TRADE_TYPE:
|
|
hash = augmentHash( hash, (XP_U8*)&entry->u.trade,
|
|
sizeof(entry->u.trade) );
|
|
break;
|
|
case PHONY_TYPE:
|
|
hash = augmentHash( hash, (XP_U8*)&entry->u.phony,
|
|
sizeof(entry->u.phony) );
|
|
break;
|
|
}
|
|
return hash;
|
|
}
|
|
|
|
XP_U32
|
|
stack_getHashOld( StackCtxt* stack )
|
|
{
|
|
XP_U16 nn, nEntries = stack->nEntries;
|
|
XP_U32 hash = 0L;
|
|
for ( nn = 0; nn < nEntries; ++nn ) {
|
|
StackEntry entry;
|
|
XP_MEMSET( &entry, 0, sizeof(entry) );
|
|
if ( !stack_getNthEntry( stack, nn, &entry ) ) {
|
|
XP_ASSERT( 0 );
|
|
}
|
|
hash = augmentFor( hash, &entry );
|
|
// XP_LOGF( "hash after %d: %.8X", nn, (unsigned int)hash );
|
|
}
|
|
XP_ASSERT( 0 != hash );
|
|
hash = finishHash( hash );
|
|
LOG_RETURNF( "%.8X", (unsigned int)hash );
|
|
return hash;
|
|
} /* stack_getHashOld */
|
|
|
|
#ifdef STREAM_VERS_HASHSTREAM
|
|
XP_U32
|
|
stack_getHash( const StackCtxt* stack )
|
|
{
|
|
XP_U32 hash;
|
|
XP_U16 len = 0;
|
|
stream_copyBits( stack->data, 0, stack->top, NULL, &len );
|
|
XP_U8 buf[len];
|
|
stream_copyBits( stack->data, 0, stack->top, buf, &len );
|
|
// LOG_HEX( buf, len, __func__ );
|
|
hash = finishHash( augmentHash( 0L, buf, len ) );
|
|
// LOG_RETURNF( "%.8X", (unsigned int)hash );
|
|
return hash;
|
|
} /* stack_getHash */
|
|
#endif
|
|
|
|
void
|
|
stack_setBitsPerTile( StackCtxt* stack, XP_U16 bitsPerTile )
|
|
{
|
|
XP_ASSERT( !!stack );
|
|
XP_ASSERT( bitsPerTile == 5 || bitsPerTile == 6 );
|
|
stack->bitsPerTile = bitsPerTile;
|
|
}
|
|
|
|
StackCtxt*
|
|
stack_make( MPFORMAL VTableMgr* vtmgr )
|
|
{
|
|
StackCtxt* result = (StackCtxt*)XP_MALLOC( mpool, sizeof( *result ) );
|
|
if ( !!result ) {
|
|
XP_MEMSET( result, 0, sizeof(*result) );
|
|
MPASSIGN(result->mpool, mpool);
|
|
result->vtmgr = vtmgr;
|
|
}
|
|
|
|
return result;
|
|
} /* stack_make */
|
|
|
|
void
|
|
stack_destroy( StackCtxt* stack )
|
|
{
|
|
if ( !!stack->data ) {
|
|
stream_destroy( stack->data );
|
|
}
|
|
/* Ok to close with a dirty stack, e.g. if not saving a deleted game */
|
|
// ASSERT_NOT_DIRTY( stack );
|
|
XP_FREE( stack->mpool, stack );
|
|
} /* stack_destroy */
|
|
|
|
void
|
|
stack_loadFromStream( StackCtxt* stack, XWStreamCtxt* stream )
|
|
{
|
|
XP_U16 nBytes = stream_getU16( stream );
|
|
|
|
if ( nBytes > 0 ) {
|
|
stack->highWaterMark = stream_getU16( stream );
|
|
stack->nEntries = stream_getU16( stream );
|
|
stack->top = stream_getU32( stream );
|
|
stack->data = mem_stream_make( MPPARM(stack->mpool) stack->vtmgr,
|
|
NULL, 0,
|
|
(MemStreamCloseCallback)NULL );
|
|
|
|
stream_getFromStream( stack->data, stream, nBytes );
|
|
} else {
|
|
XP_ASSERT( stack->nEntries == 0 );
|
|
XP_ASSERT( stack->top == 0 );
|
|
}
|
|
CLEAR_DIRTY( stack );
|
|
} /* stack_makeFromStream */
|
|
|
|
void
|
|
stack_writeToStream( const StackCtxt* stack, XWStreamCtxt* stream )
|
|
{
|
|
XP_U16 nBytes;
|
|
XWStreamCtxt* data = stack->data;
|
|
XWStreamPos oldPos = START_OF_STREAM;
|
|
|
|
if ( !!data ) {
|
|
oldPos = stream_setPos( data, POS_READ, START_OF_STREAM );
|
|
nBytes = stream_getSize( data );
|
|
} else {
|
|
nBytes = 0;
|
|
}
|
|
|
|
stream_putU16( stream, nBytes );
|
|
|
|
if ( nBytes > 0 ) {
|
|
stream_putU16( stream, stack->highWaterMark );
|
|
stream_putU16( stream, stack->nEntries );
|
|
stream_putU32( stream, stack->top );
|
|
|
|
stream_getFromStream( stream, data, nBytes );
|
|
/* in case it'll be used further */
|
|
(void)stream_setPos( data, POS_READ, oldPos );
|
|
}
|
|
CLEAR_DIRTY( stack );
|
|
} /* stack_writeToStream */
|
|
|
|
StackCtxt*
|
|
stack_copy( const StackCtxt* stack )
|
|
{
|
|
StackCtxt* newStack = NULL;
|
|
XWStreamCtxt* stream = mem_stream_make( MPPARM(stack->mpool)
|
|
stack->vtmgr, NULL, 0, NULL );
|
|
stack_writeToStream( stack, stream );
|
|
|
|
newStack = stack_make( MPPARM(stack->mpool) stack->vtmgr );
|
|
stack_loadFromStream( newStack, stream );
|
|
stack_setBitsPerTile( newStack, stack->bitsPerTile );
|
|
stream_destroy( stream );
|
|
return newStack;
|
|
}
|
|
|
|
static void
|
|
pushEntry( StackCtxt* stack, const StackEntry* entry )
|
|
{
|
|
XP_U16 ii, bitsPerTile;
|
|
XWStreamPos oldLoc;
|
|
XP_U16 nTiles = entry->u.move.moveInfo.nTiles;
|
|
XWStreamCtxt* stream = stack->data;
|
|
|
|
if ( !stream ) {
|
|
stream = mem_stream_make( MPPARM(stack->mpool) stack->vtmgr, NULL, 0,
|
|
(MemStreamCloseCallback)NULL );
|
|
stack->data = stream;
|
|
}
|
|
|
|
oldLoc = stream_setPos( stream, POS_WRITE, stack->top );
|
|
|
|
stream_putBits( stream, 2, entry->moveType );
|
|
stream_putBits( stream, 2, entry->playerNum );
|
|
|
|
switch( entry->moveType ) {
|
|
case MOVE_TYPE:
|
|
case PHONY_TYPE:
|
|
|
|
stream_putBits( stream, NTILES_NBITS, nTiles );
|
|
stream_putBits( stream, 5, entry->u.move.moveInfo.commonCoord );
|
|
stream_putBits( stream, 1, entry->u.move.moveInfo.isHorizontal );
|
|
bitsPerTile = stack->bitsPerTile;
|
|
XP_ASSERT( bitsPerTile == 5 || bitsPerTile == 6 );
|
|
for ( ii = 0; ii < nTiles; ++ii ) {
|
|
Tile tile;
|
|
stream_putBits( stream, 5,
|
|
entry->u.move.moveInfo.tiles[ii].varCoord );
|
|
|
|
tile = entry->u.move.moveInfo.tiles[ii].tile;
|
|
stream_putBits( stream, bitsPerTile, tile & TILE_VALUE_MASK );
|
|
stream_putBits( stream, 1, (tile & TILE_BLANK_BIT) != 0 );
|
|
}
|
|
if ( entry->moveType == MOVE_TYPE ) {
|
|
traySetToStream( stream, &entry->u.move.newTiles );
|
|
}
|
|
break;
|
|
|
|
case ASSIGN_TYPE:
|
|
traySetToStream( stream, &entry->u.assign.tiles );
|
|
break;
|
|
|
|
case TRADE_TYPE:
|
|
XP_ASSERT( entry->u.trade.newTiles.nTiles
|
|
== entry->u.trade.oldTiles.nTiles );
|
|
traySetToStream( stream, &entry->u.trade.oldTiles );
|
|
/* could save three bits per trade by just writing the tiles of the
|
|
second guy */
|
|
traySetToStream( stream, &entry->u.trade.newTiles );
|
|
break;
|
|
}
|
|
|
|
++stack->nEntries;
|
|
stack->highWaterMark = stack->nEntries;
|
|
stack->top = stream_setPos( stream, POS_WRITE, oldLoc );
|
|
// XP_LOGSTREAM( stack->data );
|
|
SET_DIRTY( stack );
|
|
} /* pushEntry */
|
|
|
|
static void
|
|
readEntry( const StackCtxt* stack, StackEntry* entry )
|
|
{
|
|
XP_U16 nTiles, ii, bitsPerTile;
|
|
XWStreamCtxt* stream = stack->data;
|
|
|
|
entry->moveType = (StackMoveType)stream_getBits( stream, 2 );
|
|
entry->playerNum = (XP_U8)stream_getBits( stream, 2 );
|
|
|
|
switch( entry->moveType ) {
|
|
|
|
case MOVE_TYPE:
|
|
case PHONY_TYPE:
|
|
nTiles = entry->u.move.moveInfo.nTiles =
|
|
(XP_U8)stream_getBits( stream, NTILES_NBITS );
|
|
XP_ASSERT( nTiles <= MAX_TRAY_TILES );
|
|
entry->u.move.moveInfo.commonCoord = (XP_U8)stream_getBits(stream, 5);
|
|
entry->u.move.moveInfo.isHorizontal = (XP_U8)stream_getBits(stream, 1);
|
|
bitsPerTile = stack->bitsPerTile;
|
|
XP_ASSERT( bitsPerTile == 5 || bitsPerTile == 6 );
|
|
for ( ii = 0; ii < nTiles; ++ii ) {
|
|
Tile tile;
|
|
entry->u.move.moveInfo.tiles[ii].varCoord =
|
|
(XP_U8)stream_getBits(stream, 5);
|
|
tile = (Tile)stream_getBits( stream, bitsPerTile );
|
|
if ( 0 != stream_getBits( stream, 1 ) ) {
|
|
tile |= TILE_BLANK_BIT;
|
|
}
|
|
entry->u.move.moveInfo.tiles[ii].tile = tile;
|
|
}
|
|
|
|
if ( entry->moveType == MOVE_TYPE ) {
|
|
traySetFromStream( stream, &entry->u.move.newTiles );
|
|
}
|
|
break;
|
|
|
|
case ASSIGN_TYPE:
|
|
traySetFromStream( stream, &entry->u.assign.tiles );
|
|
break;
|
|
|
|
case TRADE_TYPE:
|
|
traySetFromStream( stream, &entry->u.trade.oldTiles );
|
|
traySetFromStream( stream, &entry->u.trade.newTiles );
|
|
XP_ASSERT( entry->u.trade.newTiles.nTiles
|
|
== entry->u.trade.oldTiles.nTiles );
|
|
break;
|
|
}
|
|
|
|
} /* readEntry */
|
|
|
|
void
|
|
stack_addMove( StackCtxt* stack, XP_U16 turn, const MoveInfo* moveInfo,
|
|
const TrayTileSet* newTiles )
|
|
{
|
|
StackEntry move;
|
|
|
|
move.playerNum = (XP_U8)turn;
|
|
move.moveType = MOVE_TYPE;
|
|
|
|
XP_MEMCPY( &move.u.move.moveInfo, moveInfo, sizeof(move.u.move.moveInfo));
|
|
move.u.move.newTiles = *newTiles;
|
|
|
|
pushEntry( stack, &move );
|
|
} /* stack_addMove */
|
|
|
|
void
|
|
stack_addPhony( StackCtxt* stack, XP_U16 turn, const MoveInfo* moveInfo )
|
|
{
|
|
StackEntry move;
|
|
|
|
move.playerNum = (XP_U8)turn;
|
|
move.moveType = PHONY_TYPE;
|
|
|
|
XP_MEMCPY( &move.u.phony.moveInfo, moveInfo,
|
|
sizeof(move.u.phony.moveInfo));
|
|
|
|
pushEntry( stack, &move );
|
|
} /* stack_addPhony */
|
|
|
|
void
|
|
stack_addTrade( StackCtxt* stack, XP_U16 turn,
|
|
const TrayTileSet* oldTiles, const TrayTileSet* newTiles )
|
|
{
|
|
StackEntry move;
|
|
|
|
move.playerNum = (XP_U8)turn;
|
|
move.moveType = TRADE_TYPE;
|
|
|
|
move.u.trade.oldTiles = *oldTiles;
|
|
move.u.trade.newTiles = *newTiles;
|
|
|
|
pushEntry( stack, &move );
|
|
} /* stack_addTrade */
|
|
|
|
void
|
|
stack_addAssign( StackCtxt* stack, XP_U16 turn, const TrayTileSet* tiles )
|
|
{
|
|
StackEntry move;
|
|
|
|
move.playerNum = (XP_U8)turn;
|
|
move.moveType = ASSIGN_TYPE;
|
|
|
|
move.u.assign.tiles = *tiles;
|
|
|
|
pushEntry( stack, &move );
|
|
} /* stack_addAssign */
|
|
|
|
static XP_Bool
|
|
setCacheReadyFor( StackCtxt* stack, XP_U16 nn )
|
|
{
|
|
XP_U16 ii;
|
|
|
|
stream_setPos( stack->data, POS_READ, START_OF_STREAM );
|
|
for ( ii = 0; ii < nn; ++ii ) {
|
|
StackEntry dummy;
|
|
readEntry( stack, &dummy );
|
|
}
|
|
|
|
stack->cacheNext = nn;
|
|
stack->cachedPos = stream_getPos( stack->data, POS_READ );
|
|
|
|
return XP_TRUE;
|
|
} /* setCacheReadyFor */
|
|
|
|
XP_U16
|
|
stack_getNEntries( const StackCtxt* stack )
|
|
{
|
|
return stack->nEntries;
|
|
} /* stack_getNEntries */
|
|
|
|
XP_Bool
|
|
stack_getNthEntry( StackCtxt* stack, XP_U16 nn, StackEntry* entry )
|
|
{
|
|
XP_Bool found;
|
|
|
|
if ( nn >= stack->nEntries ) {
|
|
found = XP_FALSE;
|
|
} else if ( stack->cacheNext != nn ) {
|
|
XP_ASSERT( !!stack->data );
|
|
found = setCacheReadyFor( stack, nn );
|
|
XP_ASSERT( stack->cacheNext == nn );
|
|
} else {
|
|
found = XP_TRUE;
|
|
}
|
|
|
|
if ( found ) {
|
|
XWStreamPos oldPos = stream_setPos( stack->data, POS_READ,
|
|
stack->cachedPos );
|
|
|
|
readEntry( stack, entry );
|
|
entry->moveNum = (XP_U8)nn;
|
|
|
|
stack->cachedPos = stream_setPos( stack->data, POS_READ, oldPos );
|
|
++stack->cacheNext;
|
|
}
|
|
|
|
return found;
|
|
} /* stack_getNthEntry */
|
|
|
|
XP_Bool
|
|
stack_popEntry( StackCtxt* stack, StackEntry* entry )
|
|
{
|
|
XP_U16 nn = stack->nEntries - 1;
|
|
XP_Bool found = stack_getNthEntry( stack, nn, entry );
|
|
if ( found ) {
|
|
stack->nEntries = nn;
|
|
|
|
setCacheReadyFor( stack, nn ); /* set cachedPos by side-effect */
|
|
stack->top = stack->cachedPos;
|
|
}
|
|
// XP_LOGSTREAM( stack->data );
|
|
return found;
|
|
} /* stack_popEntry */
|
|
|
|
XP_Bool
|
|
stack_redo( StackCtxt* stack, StackEntry* entry )
|
|
{
|
|
XP_Bool canRedo = (stack->nEntries + 1) <= stack->highWaterMark;
|
|
if ( canRedo ) {
|
|
++stack->nEntries;
|
|
if ( NULL != entry ) {
|
|
stack_getNthEntry( stack, stack->nEntries-1, entry );
|
|
}
|
|
setCacheReadyFor( stack, stack->nEntries );
|
|
stack->top = stack->cachedPos;
|
|
}
|
|
return canRedo;
|
|
} /* stack_redo */
|
|
|
|
#ifdef CPLUS
|
|
}
|
|
#endif
|