mirror of
git://xwords.git.sourceforge.net/gitroot/xwords/xwords
synced 2024-12-30 10:26:58 +01:00
add jni methods, and modify common code, so a CurGameInfo struct alone
can be read from and stored to a game file.
This commit is contained in:
parent
a3a313fdb4
commit
3beba51b6f
3 changed files with 144 additions and 58 deletions
|
@ -18,6 +18,10 @@ public class XwJNI {
|
||||||
public static native boolean timerFired( int gamePtr, int why,
|
public static native boolean timerFired( int gamePtr, int why,
|
||||||
int when, int handle );
|
int when, int handle );
|
||||||
|
|
||||||
|
// Stateless methods
|
||||||
|
public static native byte[] gi_to_stream( CurGameInfo gi );
|
||||||
|
public static native void gi_from_stream( CurGameInfo gi, byte[] stream );
|
||||||
|
|
||||||
// Game methods
|
// Game methods
|
||||||
public static native int initJNI();
|
public static native int initJNI();
|
||||||
public static native void game_makeNewGame( int gamePtr,
|
public static native void game_makeNewGame( int gamePtr,
|
||||||
|
|
|
@ -132,6 +132,77 @@ loadCommonPrefs( JNIEnv* env, CommonPrefs* cp, jobject j_cp )
|
||||||
return success;
|
return success;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/****************************************************
|
||||||
|
* These two methods are stateless: no gamePtr
|
||||||
|
****************************************************/
|
||||||
|
JNIEXPORT jbyteArray JNICALL
|
||||||
|
Java_org_eehouse_android_xw4_jni_XwJNI_gi_1to_1stream
|
||||||
|
(JNIEnv* env, jclass C, jobject jgi )
|
||||||
|
{
|
||||||
|
LOG_FUNC();
|
||||||
|
jbyteArray result;
|
||||||
|
#ifdef MEM_DEBUG
|
||||||
|
MemPoolCtx* mpool = mpool_make();
|
||||||
|
#endif
|
||||||
|
CurGameInfo* gi = makeGI( MPPARM(mpool) env, jgi );
|
||||||
|
VTableMgr* vtMgr = make_vtablemgr( MPPARM_NOCOMMA(mpool) );
|
||||||
|
XWStreamCtxt* stream = mem_stream_make( MPPARM(mpool) vtMgr,
|
||||||
|
NULL, 0, NULL );
|
||||||
|
|
||||||
|
game_saveToStream( NULL, gi, stream );
|
||||||
|
destroyGI( MPPARM(mpool) gi );
|
||||||
|
|
||||||
|
int nBytes = stream_getSize( stream );
|
||||||
|
result = (*env)->NewByteArray( env, nBytes );
|
||||||
|
jbyte* jelems = (*env)->GetByteArrayElements( env, result, NULL );
|
||||||
|
stream_getBytes( stream, jelems, nBytes );
|
||||||
|
(*env)->ReleaseByteArrayElements( env, result, jelems, 0 );
|
||||||
|
stream_destroy( stream );
|
||||||
|
|
||||||
|
vtmgr_destroy( MPPARM(mpool) vtMgr );
|
||||||
|
#ifdef MEM_DEBUG
|
||||||
|
mpool_destroy( mpool );
|
||||||
|
#endif
|
||||||
|
LOG_RETURN_VOID();
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
|
||||||
|
JNIEXPORT void JNICALL
|
||||||
|
Java_org_eehouse_android_xw4_jni_XwJNI_gi_1from_1stream
|
||||||
|
( JNIEnv* env, jclass C, jobject jgi, jbyteArray jstream )
|
||||||
|
{
|
||||||
|
LOG_FUNC();
|
||||||
|
#ifdef MEM_DEBUG
|
||||||
|
MemPoolCtx* mpool = mpool_make();
|
||||||
|
#endif
|
||||||
|
VTableMgr* vtMgr = make_vtablemgr( MPPARM_NOCOMMA(mpool) );
|
||||||
|
|
||||||
|
jbyte* jelems = (*env)->GetByteArrayElements( env, jstream, NULL );
|
||||||
|
XWStreamCtxt* stream = mem_stream_make( MPPARM(mpool) vtMgr,
|
||||||
|
NULL, 0, NULL );
|
||||||
|
int len = (*env)->GetArrayLength( env, jstream );
|
||||||
|
stream_putBytes( stream, jelems, len );
|
||||||
|
(*env)->ReleaseByteArrayElements( env, jstream, jelems, 0 );
|
||||||
|
|
||||||
|
CurGameInfo gi;
|
||||||
|
XP_MEMSET( &gi, 0, sizeof(gi) );
|
||||||
|
if ( game_makeFromStream( MPPARM(mpool) stream, NULL,
|
||||||
|
&gi, NULL, NULL, NULL, NULL, NULL ) ) {
|
||||||
|
setJGI( env, jgi, &gi );
|
||||||
|
} else {
|
||||||
|
XP_LOGF( "%s: game_makeFromStream failed", __func__ );
|
||||||
|
}
|
||||||
|
|
||||||
|
gi_disposePlayerInfo( MPPARM(mpool) &gi );
|
||||||
|
|
||||||
|
stream_destroy( stream );
|
||||||
|
vtmgr_destroy( MPPARM(mpool) vtMgr );
|
||||||
|
#ifdef MEM_DEBUG
|
||||||
|
mpool_destroy( mpool );
|
||||||
|
#endif
|
||||||
|
LOG_RETURN_VOID();
|
||||||
|
}
|
||||||
|
|
||||||
typedef struct _JNIState {
|
typedef struct _JNIState {
|
||||||
XWGame game;
|
XWGame game;
|
||||||
JNIEnv* env;
|
JNIEnv* env;
|
||||||
|
@ -140,12 +211,12 @@ typedef struct _JNIState {
|
||||||
} JNIState;
|
} JNIState;
|
||||||
|
|
||||||
#define XWJNI_START() { \
|
#define XWJNI_START() { \
|
||||||
|
XP_LOGF( "%s(env=%x)", __func__, env ); \
|
||||||
XP_ASSERT( 0 != gamePtr ); \
|
XP_ASSERT( 0 != gamePtr ); \
|
||||||
JNIState* state = (JNIState*)gamePtr; \
|
JNIState* state = (JNIState*)gamePtr; \
|
||||||
MPSLOT; \
|
MPSLOT; \
|
||||||
MPASSIGN( mpool, state->mpool); \
|
MPASSIGN( mpool, state->mpool); \
|
||||||
AndGlobals* globals = &state->globals; \
|
AndGlobals* globals = &state->globals; \
|
||||||
LOG_FUNC(); \
|
|
||||||
/* if reentrant must be from same thread */ \
|
/* if reentrant must be from same thread */ \
|
||||||
XP_ASSERT( state->env == 0 || state->env == env ); \
|
XP_ASSERT( state->env == 0 || state->env == env ); \
|
||||||
JNIEnv* _oldEnv = state->env; \
|
JNIEnv* _oldEnv = state->env; \
|
||||||
|
@ -183,16 +254,13 @@ Java_org_eehouse_android_xw4_jni_XwJNI_game_1makeNewGame
|
||||||
XWJNI_START();
|
XWJNI_START();
|
||||||
CurGameInfo* gi = makeGI( MPPARM(mpool) env, j_gi );
|
CurGameInfo* gi = makeGI( MPPARM(mpool) env, j_gi );
|
||||||
globals->gi = gi;
|
globals->gi = gi;
|
||||||
XW_UtilCtxt* util = makeUtil( MPPARM(mpool) &state->env, j_util, gi, globals );
|
XW_UtilCtxt* util = makeUtil( MPPARM(mpool) &state->env, j_util, gi,
|
||||||
|
globals );
|
||||||
globals->util = util;
|
globals->util = util;
|
||||||
DrawCtx* dctx = makeDraw( MPPARM(mpool) &state->env, j_draw );
|
DrawCtx* dctx = makeDraw( MPPARM(mpool) &state->env, j_draw );
|
||||||
globals->dctx = dctx;
|
globals->dctx = dctx;
|
||||||
CommonPrefs cp;
|
CommonPrefs cp;
|
||||||
(void)loadCommonPrefs( env, &cp, j_cp );
|
(void)loadCommonPrefs( env, &cp, j_cp );
|
||||||
#ifndef XWFEATURE_STANDALONE_ONLY
|
|
||||||
/* TransportProcs proc; */
|
|
||||||
/* loadTransportProcs( &procs, j_procs ); */
|
|
||||||
#endif
|
|
||||||
|
|
||||||
XP_LOGF( "calling game_makeNewGame" );
|
XP_LOGF( "calling game_makeNewGame" );
|
||||||
game_makeNewGame( MPPARM(mpool) &state->game, gi, util, dctx, gameID,
|
game_makeNewGame( MPPARM(mpool) &state->game, gi, util, dctx, gameID,
|
||||||
|
@ -234,7 +302,7 @@ JNIEXPORT void JNICALL Java_org_eehouse_android_xw4_jni_XwJNI_game_1dispose
|
||||||
XP_FREE( mpool, state );
|
XP_FREE( mpool, state );
|
||||||
mpool_destroy( mpool );
|
mpool_destroy( mpool );
|
||||||
LOG_RETURN_VOID();
|
LOG_RETURN_VOID();
|
||||||
}
|
} /* game_dispose */
|
||||||
|
|
||||||
JNIEXPORT jboolean JNICALL
|
JNIEXPORT jboolean JNICALL
|
||||||
Java_org_eehouse_android_xw4_jni_XwJNI_game_1makeFromStream
|
Java_org_eehouse_android_xw4_jni_XwJNI_game_1makeFromStream
|
||||||
|
@ -254,7 +322,6 @@ Java_org_eehouse_android_xw4_jni_XwJNI_game_1makeFromStream
|
||||||
XWStreamCtxt* stream = mem_stream_make( MPPARM(mpool) globals->vtMgr,
|
XWStreamCtxt* stream = mem_stream_make( MPPARM(mpool) globals->vtMgr,
|
||||||
NULL, 0, NULL );
|
NULL, 0, NULL );
|
||||||
int len = (*env)->GetArrayLength( env, jstream );
|
int len = (*env)->GetArrayLength( env, jstream );
|
||||||
XP_LOGF( "putting %d bytes into stream", len );
|
|
||||||
stream_putBytes( stream, jelems, len );
|
stream_putBytes( stream, jelems, len );
|
||||||
(*env)->ReleaseByteArrayElements( env, jstream, jelems, 0 );
|
(*env)->ReleaseByteArrayElements( env, jstream, jelems, 0 );
|
||||||
|
|
||||||
|
@ -266,7 +333,11 @@ Java_org_eehouse_android_xw4_jni_XwJNI_game_1makeFromStream
|
||||||
NULL );
|
NULL );
|
||||||
stream_destroy( stream );
|
stream_destroy( stream );
|
||||||
|
|
||||||
setJGI( env, jgi, globals->gi );
|
if ( result ) {
|
||||||
|
setJGI( env, jgi, globals->gi );
|
||||||
|
} else {
|
||||||
|
XP_LOGF( "%s: need to free stuff allocated above", __func__ );
|
||||||
|
}
|
||||||
|
|
||||||
XWJNI_END();
|
XWJNI_END();
|
||||||
return result;
|
return result;
|
||||||
|
|
|
@ -179,48 +179,57 @@ game_makeFromStream( MPFORMAL XWStreamCtxt* stream, XWGame* game,
|
||||||
strVersion = stream_getU8( stream );
|
strVersion = stream_getU8( stream );
|
||||||
XP_DEBUGF( "strVersion = %d", (XP_U16)strVersion );
|
XP_DEBUGF( "strVersion = %d", (XP_U16)strVersion );
|
||||||
|
|
||||||
if ( strVersion <= CUR_STREAM_VERS ) {
|
if ( strVersion > CUR_STREAM_VERS ) {
|
||||||
stream_setVersion( stream, strVersion );
|
|
||||||
|
|
||||||
gi_readFromStream( MPPARM(mpool) stream, gi );
|
|
||||||
|
|
||||||
/* 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
|
|
||||||
) {
|
|
||||||
hasComms = stream_getU8( stream );
|
|
||||||
}
|
|
||||||
|
|
||||||
if ( hasComms ) {
|
|
||||||
game->comms = comms_makeFromStream( MPPARM(mpool) stream, util,
|
|
||||||
procs );
|
|
||||||
} else {
|
|
||||||
game->comms = NULL;
|
|
||||||
}
|
|
||||||
|
|
||||||
XP_ASSERT( !!dict );
|
|
||||||
game->model = model_makeFromStream( MPPARM(mpool) stream, dict, util );
|
|
||||||
|
|
||||||
game->server = server_makeFromStream( MPPARM(mpool) stream,
|
|
||||||
game->model, game->comms,
|
|
||||||
util, gi->nPlayers );
|
|
||||||
|
|
||||||
game->board = board_makeFromStream( MPPARM(mpool) stream, game->model,
|
|
||||||
game->server, draw, util,
|
|
||||||
gi->nPlayers );
|
|
||||||
server_prefsChanged( game->server, cp );
|
|
||||||
board_prefsChanged( game->board, cp );
|
|
||||||
draw_dictChanged( draw, dict );
|
|
||||||
success = XP_TRUE;
|
|
||||||
} else {
|
|
||||||
XP_LOGF( "%s: aborting; stream version too new!", __func__ );
|
XP_LOGF( "%s: aborting; stream version too new!", __func__ );
|
||||||
|
} 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_LOGF( "%s: gi was all we got; failing." );
|
||||||
|
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
|
||||||
|
) {
|
||||||
|
hasComms = stream_getU8( stream );
|
||||||
|
}
|
||||||
|
|
||||||
|
if ( hasComms ) {
|
||||||
|
game->comms = comms_makeFromStream( MPPARM(mpool) stream, util,
|
||||||
|
procs );
|
||||||
|
} else {
|
||||||
|
game->comms = NULL;
|
||||||
|
}
|
||||||
|
|
||||||
|
XP_ASSERT( !!dict );
|
||||||
|
game->model = model_makeFromStream( MPPARM(mpool) stream, dict, util );
|
||||||
|
|
||||||
|
game->server = server_makeFromStream( MPPARM(mpool) stream,
|
||||||
|
game->model, game->comms,
|
||||||
|
util, gi->nPlayers );
|
||||||
|
|
||||||
|
game->board = board_makeFromStream( MPPARM(mpool) stream, game->model,
|
||||||
|
game->server, draw, util,
|
||||||
|
gi->nPlayers );
|
||||||
|
server_prefsChanged( game->server, cp );
|
||||||
|
board_prefsChanged( game->board, cp );
|
||||||
|
draw_dictChanged( draw, dict );
|
||||||
|
success = XP_TRUE;
|
||||||
|
} while( XP_FALSE );
|
||||||
}
|
}
|
||||||
return success;
|
return success;
|
||||||
} /* game_makeFromStream */
|
} /* game_makeFromStream */
|
||||||
|
@ -233,17 +242,19 @@ game_saveToStream( const XWGame* game, const CurGameInfo* gi,
|
||||||
|
|
||||||
gi_writeToStream( stream, gi );
|
gi_writeToStream( stream, gi );
|
||||||
|
|
||||||
stream_putU8( stream, (XP_U8)!!game->comms );
|
if ( !!game ) {
|
||||||
|
stream_putU8( stream, (XP_U8)!!game->comms );
|
||||||
#ifdef XWFEATURE_STANDALONE_ONLY
|
#ifdef XWFEATURE_STANDALONE_ONLY
|
||||||
XP_ASSERT( !game->comms );
|
XP_ASSERT( !game->comms );
|
||||||
#endif
|
#endif
|
||||||
if ( !!game->comms ) {
|
if ( !!game->comms ) {
|
||||||
comms_writeToStream( game->comms, stream );
|
comms_writeToStream( game->comms, stream );
|
||||||
}
|
}
|
||||||
|
|
||||||
model_writeToStream( game->model, stream );
|
model_writeToStream( game->model, stream );
|
||||||
server_writeToStream( game->server, stream );
|
server_writeToStream( game->server, stream );
|
||||||
board_writeToStream( game->board, stream );
|
board_writeToStream( game->board, stream );
|
||||||
|
}
|
||||||
} /* game_saveToStream */
|
} /* game_saveToStream */
|
||||||
|
|
||||||
void
|
void
|
||||||
|
|
Loading…
Reference in a new issue