use undo/redo rather than copying stack to figure past move score.

This may have been contributing to crashes, but the new code's more
effecient anyway.
This commit is contained in:
Andy2 2011-11-10 06:52:33 -08:00
parent 00f743409a
commit 43a79a0b47

View file

@ -835,7 +835,7 @@ model_currentMoveToStream( ModelCtxt* model, XP_S16 turn,
stream_putBits( stream, NUMCOLS_NBITS, row ); stream_putBits( stream, NUMCOLS_NBITS, row );
stream_putBits( stream, 1, isBlank ); stream_putBits( stream, 1, isBlank );
} }
} /* model_turnToStream */ } /* model_currentMoveToStream */
/* Take stream as the source of info about what tiles to move from tray to /* Take stream as the source of info about what tiles to move from tray to
* board. Undo any current move first -- a player on this device might be * board. Undo any current move first -- a player on this device might be
@ -877,15 +877,15 @@ model_makeTurnFromStream( ModelCtxt* model, XP_U16 playerNum,
foundAt = model_trayContains( model, playerNum, moveTile ); foundAt = model_trayContains( model, playerNum, moveTile );
if ( foundAt == -1 ) { if ( foundAt == -1 ) {
XP_ASSERT( EMPTY_TILE==model_getPlayerTile(model, playerNum, 0)); XP_ASSERT( EMPTY_TILE == model_getPlayerTile(model, playerNum, 0));
(void)model_removePlayerTile( model, playerNum, -1 ); (void)model_removePlayerTile( model, playerNum, -1 );
model_addPlayerTile( model, playerNum, -1, moveTile ); model_addPlayerTile( model, playerNum, -1, moveTile );
} }
model_moveTrayToBoard( model, playerNum, col, row, foundAt, tileFace); model_moveTrayToBoard( model, playerNum, col, row, foundAt, tileFace );
} }
} /* model_makeMoveFromStream */ } /* model_makeTurnFromStream */
void void
model_makeTurnFromMoveInfo( ModelCtxt* model, XP_U16 playerNum, model_makeTurnFromMoveInfo( ModelCtxt* model, XP_U16 playerNum,
@ -1546,7 +1546,7 @@ model_getPlayerTiles( const ModelCtxt* model, XP_S16 turn )
} /* model_getPlayerTile */ } /* model_getPlayerTile */
static void static void
addPlayerTile( ModelCtxt* model, XP_S16 turn, XP_S16 index, Tile tile ) addPlayerTile( ModelCtxt* model, XP_S16 turn, XP_S16 index, const Tile tile )
{ {
PlayerCtxt* player; PlayerCtxt* player;
short ii; short ii;
@ -1892,19 +1892,6 @@ printMovePost( ModelCtxt* model, XP_U16 XP_UNUSED(moveN),
printString( stream, (XP_UCHAR*)XP_CR ); printString( stream, (XP_UCHAR*)XP_CR );
} /* printMovePost */ } /* printMovePost */
static void
copyStack( ModelCtxt* model, StackCtxt* destStack, const StackCtxt* srcStack )
{
XWStreamCtxt* stream = mem_stream_make( MPPARM(model->vol.mpool)
util_getVTManager(model->vol.util),
NULL, 0, NULL );
stack_writeToStream( (StackCtxt*)srcStack, stream );
stack_loadFromStream( destStack, stream );
stream_destroy( stream );
} /* copyStack */
static ModelCtxt* static ModelCtxt*
makeTmpModel( ModelCtxt* model, XWStreamCtxt* stream, makeTmpModel( ModelCtxt* model, XWStreamCtxt* stream,
MovePrintFuncPre mpf_pre, MovePrintFuncPost mpf_post, MovePrintFuncPre mpf_pre, MovePrintFuncPost mpf_post,
@ -1952,11 +1939,11 @@ typedef struct _FirstWordData {
static XP_Bool static XP_Bool
getFirstWord( const XP_UCHAR* word, XP_Bool isLegal, getFirstWord( const XP_UCHAR* word, XP_Bool isLegal,
#ifdef XWFEATURE_BOARDWORDS #ifdef XWFEATURE_BOARDWORDS
const MoveInfo* XP_UNUSED(movei), XP_U16 XP_UNUSED(start), XP_U16 XP_UNUSED(end), const MoveInfo* XP_UNUSED(movei), XP_U16 XP_UNUSED(start),
XP_U16 XP_UNUSED(end),
#endif #endif
void* closure ) void* closure )
{ {
LOG_FUNC();
if ( isLegal ) { if ( isLegal ) {
FirstWordData* data = (FirstWordData*)closure; FirstWordData* data = (FirstWordData*)closure;
if ( '\0' == data->word[0] && '\0' != word[0] ) { if ( '\0' == data->word[0] && '\0' != word[0] ) {
@ -1966,6 +1953,21 @@ getFirstWord( const XP_UCHAR* word, XP_Bool isLegal,
return XP_TRUE; return XP_TRUE;
} }
static void
redoEntries( ModelCtxt* model, StackCtxt* stack, XP_U16 nBefore,
XP_U16 nAfter, WordNotifierInfo* ni )
{
while ( nAfter < nBefore ) {
StackEntry entry;
if ( ! stack_redo( stack, &entry ) ) {
XP_ASSERT( 0 );
break;
}
modelAddEntry( model, nAfter++, &entry, XP_FALSE, NULL, ni,
NULL, NULL, NULL );
}
}
static void static void
scoreLastMove( ModelCtxt* model, MoveInfo* moveInfo, XP_U16 howMany, scoreLastMove( ModelCtxt* model, MoveInfo* moveInfo, XP_U16 howMany,
XP_UCHAR* buf, XP_U16* bufLen ) XP_UCHAR* buf, XP_U16* bufLen )
@ -1981,28 +1983,29 @@ scoreLastMove( ModelCtxt* model, MoveInfo* moveInfo, XP_U16 howMany,
const XP_UCHAR* format; const XP_UCHAR* format;
WordNotifierInfo notifyInfo; WordNotifierInfo notifyInfo;
FirstWordData data; FirstWordData data;
StackCtxt* stack = model->vol.stack;
ModelCtxt* tmpModel = makeTmpModel( model, NULL, NULL, NULL, NULL ); XP_U16 nEntriesBefore = stack_getNEntries( stack );
XP_U16 nEntriesAfter;
XP_U16 turn; XP_U16 turn;
XP_S16 moveNum = -1; XP_S16 moveNum = -1;
copyStack( model, tmpModel->vol.stack, model->vol.stack );
if ( !model_undoLatestMoves( tmpModel, NULL, howMany, &turn, if ( !model_undoLatestMoves( model, NULL, howMany,
&moveNum ) ) { &turn, &moveNum ) ) {
XP_ASSERT( 0 ); XP_ASSERT( 0 );
} }
nEntriesAfter = stack_getNEntries( stack );
data.word[0] = '\0'; data.word[0] = '\0';
notifyInfo.proc = getFirstWord; notifyInfo.proc = getFirstWord;
notifyInfo.closure = &data; notifyInfo.closure = &data;
score = figureMoveScore( tmpModel, turn, moveInfo, (EngineCtxt*)NULL, score = figureMoveScore( model, turn, moveInfo, (EngineCtxt*)NULL,
(XWStreamCtxt*)NULL, &notifyInfo ); (XWStreamCtxt*)NULL, &notifyInfo );
model_destroy( tmpModel );
format = util_getUserString( model->vol.util, STRSD_SUMMARYSCORED ); format = util_getUserString( model->vol.util, STRSD_SUMMARYSCORED );
*bufLen = XP_SNPRINTF( buf, *bufLen, format, data.word, score ); *bufLen = XP_SNPRINTF( buf, *bufLen, format, data.word, score );
redoEntries( model, stack, nEntriesBefore, nEntriesAfter, NULL );
} }
} /* scoreLastMove */ } /* scoreLastMove */
@ -2131,15 +2134,7 @@ model_listWordsThrough( ModelCtxt* model, XP_U16 col, XP_U16 row,
/* Now push the undone moves back into the model one at a time. /* Now push the undone moves back into the model one at a time.
recordWord() will add each played word to the stream as it's recordWord() will add each played word to the stream as it's
scored */ scored */
while ( nEntriesAfter < nEntriesBefore ) { redoEntries( model, stack, nEntriesBefore, nEntriesAfter, &ni );
StackEntry entry;
if ( ! stack_redo( stack, &entry ) ) {
XP_ASSERT( 0 );
break;
}
modelAddEntry( model, nEntriesAfter++, &entry, XP_FALSE, NULL, &ni,
NULL, NULL, NULL );
}
} }
} /* model_listWordsThrough */ } /* model_listWordsThrough */
#endif #endif