mirror of
git://xwords.git.sourceforge.net/gitroot/xwords/xwords
synced 2025-01-18 22:26:30 +01:00
3923a2d9a6
Add dutil proc haveGame() and use it to detect duplicate invitations. I'm passing, but ignoring on android, the channel, which means that for now you can't invite yourself and on-device testing requires having CrossWords and CrossDbg or a second user.
1162 lines
36 KiB
C
1162 lines
36 KiB
C
/* -*- compile-command: "find-and-gradle.sh inXw4dDeb"; -*- */
|
|
/*
|
|
* Copyright 2001 - 2022 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 <sys/time.h>
|
|
|
|
#include <jni.h>
|
|
|
|
#include "comtypes.h"
|
|
#include "utilwrapper.h"
|
|
#include "anddict.h"
|
|
#include "andutils.h"
|
|
#include "paths.h"
|
|
#include "LocalizedStrIncludes.h"
|
|
#include "dbgutil.h"
|
|
#include "nli.h"
|
|
#include "strutils.h"
|
|
|
|
#define MAX_QUANTITY_STRS 4
|
|
|
|
typedef struct _AndDUtil {
|
|
XW_DUtilCtxt dutil;
|
|
JNIUtilCtxt* jniutil;
|
|
jobject jdutil; /* global ref to object implementing XW_DUtilCtxt */
|
|
DictMgrCtxt* dictMgr;
|
|
XP_UCHAR* userStrings[N_AND_USER_STRINGS];
|
|
XP_U32 userStringsBits;
|
|
#ifdef MAP_THREAD_TO_ENV
|
|
EnvThreadInfo* ti;
|
|
#endif
|
|
#ifdef XWFEATURE_DEVID
|
|
XP_UCHAR* devIDStorage;
|
|
#endif
|
|
} AndDUtil;
|
|
|
|
typedef struct _TimerStorage {
|
|
XWTimerProc proc;
|
|
void* closure;
|
|
} TimerStorage;
|
|
|
|
typedef struct _AndUtil {
|
|
XW_UtilCtxt util;
|
|
#ifdef MAP_THREAD_TO_ENV
|
|
EnvThreadInfo* ti;
|
|
#endif
|
|
jobject jutil; /* global ref to object implementing XW_UtilCtxt */
|
|
TimerStorage timerStorage[NUM_TIMERS_PLUS_ONE];
|
|
} AndUtil;
|
|
|
|
#ifndef XWFEATURE_STANDALONE_ONLY
|
|
static XWStreamCtxt*
|
|
and_util_makeStreamFromAddr( XW_UtilCtxt* uc, XWEnv XP_UNUSED(xwe),
|
|
XP_PlayerAddr channelNo )
|
|
{
|
|
#ifdef DEBUG
|
|
AndUtil* util = (AndUtil*)uc;
|
|
#endif
|
|
AndGameGlobals* globals = (AndGameGlobals*)uc->closure;
|
|
XWStreamCtxt* stream = and_empty_stream( MPPARM(util->util.mpool)
|
|
globals );
|
|
stream_setAddress( stream, channelNo );
|
|
stream_setOnCloseProc( stream, and_send_on_close );
|
|
return stream;
|
|
}
|
|
#endif
|
|
|
|
#define UTIL_CBK_HEADER(nam,sig) \
|
|
AndUtil* util = (AndUtil*)uc; \
|
|
JNIEnv* env = xwe; \
|
|
ASSERT_ENV(util->ti, env); \
|
|
if ( NULL != util->jutil ) { \
|
|
jmethodID mid = getMethodID( env, util->jutil, nam, sig )
|
|
|
|
#define UTIL_CBK_TAIL() \
|
|
} else { \
|
|
XP_LOGFF( "skipping call into java because jutil==NULL" ); \
|
|
}
|
|
|
|
#define DUTIL_CBK_HEADER(nam,sig) \
|
|
AndDUtil* dutil = (AndDUtil*)duc; \
|
|
JNIEnv* env = xwe; \
|
|
ASSERT_ENV(dutil->ti, env); \
|
|
if ( NULL != dutil->jdutil ) { \
|
|
jmethodID mid = getMethodID( env, dutil->jdutil, nam, sig )
|
|
|
|
#define DUTIL_CBK_TAIL() UTIL_CBK_TAIL()
|
|
|
|
static void
|
|
and_util_userError( XW_UtilCtxt* uc, XWEnv xwe, UtilErrID id )
|
|
{
|
|
UTIL_CBK_HEADER( "userError", "(I)V" );
|
|
(*env)->CallVoidMethod( env, util->jutil, mid, id );
|
|
if ((*env)->ExceptionOccurred(env)) {
|
|
(*env)->ExceptionDescribe(env);
|
|
(*env)->ExceptionClear(env);
|
|
XP_LOGFF( "exception found" );
|
|
}
|
|
UTIL_CBK_TAIL();
|
|
}
|
|
|
|
static void
|
|
and_util_notifyMove( XW_UtilCtxt* uc, XWEnv xwe, XWStreamCtxt* stream )
|
|
{
|
|
UTIL_CBK_HEADER("notifyMove", "(Ljava/lang/String;)V" );
|
|
|
|
jstring jstr = NULL;
|
|
if ( NULL != stream ) {
|
|
jstr = streamToJString( env, stream );
|
|
}
|
|
(*env)->CallVoidMethod( env, util->jutil, mid, jstr );
|
|
deleteLocalRef( env, jstr );
|
|
UTIL_CBK_TAIL();
|
|
}
|
|
|
|
static void
|
|
and_util_notifyTrade( XW_UtilCtxt* uc, XWEnv xwe, const XP_UCHAR** tiles, XP_U16 nTiles )
|
|
{
|
|
UTIL_CBK_HEADER("notifyTrade", "([Ljava/lang/String;)V" );
|
|
jobjectArray jtiles = makeStringArray( env, nTiles, tiles );
|
|
(*env)->CallVoidMethod( env, util->jutil, mid, jtiles );
|
|
deleteLocalRef( env, jtiles );
|
|
UTIL_CBK_TAIL();
|
|
}
|
|
|
|
static void
|
|
and_util_notifyPickTileBlank( XW_UtilCtxt* uc, XWEnv xwe, XP_U16 playerNum,
|
|
XP_U16 col, XP_U16 row,
|
|
const XP_UCHAR** tileFaces, XP_U16 nTiles )
|
|
{
|
|
UTIL_CBK_HEADER("notifyPickTileBlank", "(III[Ljava/lang/String;)V" );
|
|
|
|
jobject jtexts = makeStringArray( env, nTiles, tileFaces );
|
|
|
|
(*env)->CallVoidMethod( env, util->jutil, mid, playerNum, col, row, jtexts );
|
|
|
|
deleteLocalRef( env, jtexts );
|
|
UTIL_CBK_TAIL();
|
|
}
|
|
|
|
static void
|
|
and_util_informNeedPickTiles( XW_UtilCtxt* uc, XWEnv xwe, XP_Bool isInitial,
|
|
XP_U16 player, XP_U16 nToPick,
|
|
XP_U16 nFaces, const XP_UCHAR** faces,
|
|
const XP_U16* counts )
|
|
{
|
|
UTIL_CBK_HEADER("informNeedPickTiles",
|
|
"(ZII[Ljava/lang/String;[I)V" );
|
|
jobject jtexts = makeStringArray( env, nFaces, faces );
|
|
jobject jcounts = makeIntArray( env, nFaces, counts, sizeof(counts[0]) );
|
|
|
|
(*env)->CallVoidMethod( env, util->jutil, mid, isInitial, player,
|
|
nToPick, jtexts, jcounts );
|
|
|
|
deleteLocalRefs( env, jtexts, jcounts, DELETE_NO_REF );
|
|
UTIL_CBK_TAIL();
|
|
} /* and_util_informNeedPickTiles */
|
|
|
|
static void
|
|
and_util_informNeedPassword( XW_UtilCtxt* uc, XWEnv xwe, XP_U16 player,
|
|
const XP_UCHAR* name )
|
|
{
|
|
UTIL_CBK_HEADER("informNeedPassword", "(ILjava/lang/String;)V" );
|
|
|
|
jstring jname = (*env)->NewStringUTF( env, name );
|
|
(*env)->CallVoidMethod( env, util->jutil, mid, player, jname );
|
|
deleteLocalRef( env, jname );
|
|
UTIL_CBK_TAIL();
|
|
}
|
|
|
|
static void
|
|
and_util_trayHiddenChange(XW_UtilCtxt* uc, XWEnv xwe,
|
|
XW_TrayVisState newState,
|
|
XP_U16 nVisibleRows )
|
|
{
|
|
}
|
|
|
|
static void
|
|
and_util_yOffsetChange(XW_UtilCtxt* uc, XWEnv xwe, XP_U16 maxOffset,
|
|
XP_U16 oldOffset, XP_U16 newOffset )
|
|
{
|
|
#if 0
|
|
AndUtil* util = (AndUtil*)uc;
|
|
JNIEnv* env = *util->env;
|
|
const char* sig = "(III)V";
|
|
jmethodID mid = getMethodID( env, util->jutil, "yOffsetChange", sig );
|
|
|
|
(*env)->CallVoidMethod( env, util->jutil, mid,
|
|
maxOffset, oldOffset, newOffset );
|
|
#endif
|
|
}
|
|
|
|
#ifdef XWFEATURE_TURNCHANGENOTIFY
|
|
static void
|
|
and_util_turnChanged( XW_UtilCtxt* uc, XWEnv xwe, XP_S16 turn )
|
|
{
|
|
UTIL_CBK_HEADER( "turnChanged", "(I)V" );
|
|
(*env)->CallVoidMethod( env, util->jutil, mid, turn );
|
|
UTIL_CBK_TAIL();
|
|
}
|
|
#endif
|
|
|
|
static void
|
|
and_util_informMove( XW_UtilCtxt* uc, XWEnv xwe, XP_S16 turn, XWStreamCtxt* expl,
|
|
XWStreamCtxt* words )
|
|
{
|
|
UTIL_CBK_HEADER( "informMove", "(ILjava/lang/String;Ljava/lang/String;)V" );
|
|
jstring jexpl = streamToJString( env, expl );
|
|
jstring jwords = !!words ? streamToJString( env, words ) : NULL;
|
|
(*env)->CallVoidMethod( env, util->jutil, mid, turn, jexpl, jwords );
|
|
deleteLocalRefs( env, jexpl, jwords, DELETE_NO_REF );
|
|
UTIL_CBK_TAIL();
|
|
}
|
|
|
|
static void
|
|
and_util_notifyDupStatus( XW_UtilCtxt* uc, XWEnv xwe, XP_Bool amHost, const XP_UCHAR* msg )
|
|
{
|
|
UTIL_CBK_HEADER( "notifyDupStatus", "(ZLjava/lang/String;)V" );
|
|
jstring jmsg = (*env)->NewStringUTF( env, msg );
|
|
(*env)->CallVoidMethod( env, util->jutil, mid, amHost, jmsg );
|
|
deleteLocalRefs( env, jmsg, DELETE_NO_REF );
|
|
UTIL_CBK_TAIL();
|
|
}
|
|
|
|
static void
|
|
and_util_informUndo( XW_UtilCtxt* uc, XWEnv xwe )
|
|
{
|
|
UTIL_CBK_HEADER( "informUndo", "()V" );
|
|
(*env)->CallVoidMethod( env, util->jutil, mid );
|
|
UTIL_CBK_TAIL();
|
|
}
|
|
|
|
static void
|
|
and_util_informNetDict( XW_UtilCtxt* uc, XWEnv xwe, const XP_UCHAR* isoCodeStr,
|
|
const XP_UCHAR* oldName, const XP_UCHAR* newName,
|
|
const XP_UCHAR* newSum, XWPhoniesChoice phoniesAction )
|
|
{
|
|
LOG_FUNC();
|
|
UTIL_CBK_HEADER( "informNetDict",
|
|
"(Ljava/lang/String;Ljava/lang/String;Ljava/lang/String;"
|
|
"Ljava/lang/String;L"
|
|
PKG_PATH("jni/CurGameInfo$XWPhoniesChoice") ";)V" );
|
|
jstring jIsoCode = (*env)->NewStringUTF( env, isoCodeStr );
|
|
jstring jnew = (*env)->NewStringUTF( env, newName );
|
|
jstring jsum = (*env)->NewStringUTF( env, newSum );
|
|
jstring jold = (*env)->NewStringUTF( env, oldName );
|
|
jobject jphon = intToJEnum( env, phoniesAction,
|
|
PKG_PATH("jni/CurGameInfo$XWPhoniesChoice") );
|
|
|
|
(*env)->CallVoidMethod( env, util->jutil, mid, jIsoCode, jold, jnew, jsum,
|
|
jphon );
|
|
deleteLocalRefs( env, jnew, jold, jsum, jphon, jIsoCode, DELETE_NO_REF );
|
|
|
|
UTIL_CBK_TAIL();
|
|
}
|
|
|
|
const DictionaryCtxt*
|
|
and_util_getDict( XW_UtilCtxt* uc, XWEnv xwe,
|
|
const XP_UCHAR* XP_UNUSED_DBG(isoCode),
|
|
const XP_UCHAR* dictName )
|
|
{
|
|
XP_LOGFF( "(isoCode: %s, name: %s)", isoCode, dictName );
|
|
JNIEnv* env = xwe;
|
|
XW_DUtilCtxt* duc = util_getDevUtilCtxt( uc, xwe );
|
|
|
|
DictMgrCtxt* dictMgr = ((AndDUtil*)duc)->dictMgr;
|
|
DictionaryCtxt* dict = (DictionaryCtxt*)
|
|
dmgr_get( dictMgr, xwe, dictName );
|
|
if ( !dict ) {
|
|
jstring jname = (*env)->NewStringUTF( env, dictName );
|
|
|
|
jobjectArray jstrs = makeStringArray( env, 1, NULL );
|
|
jobjectArray jbytes = makeByteArrayArray( env, 1 );
|
|
|
|
DUTIL_CBK_HEADER( "getDictPath",
|
|
"(Ljava/lang/String;[Ljava/lang/String;[[B)V" );
|
|
(*env)->CallVoidMethod( env, dutil->jdutil, mid, jname, jstrs, jbytes );
|
|
DUTIL_CBK_TAIL();
|
|
|
|
jstring jpath = (*env)->GetObjectArrayElement( env, jstrs, 0 );
|
|
jbyteArray jdata = (*env)->GetObjectArrayElement( env, jbytes, 0 );
|
|
|
|
dict = makeDict( MPPARM(duc->mpool) xwe,
|
|
TI_IF(&globalState->ti)
|
|
dictMgr, ((AndDUtil*)duc)->jniutil,
|
|
jname, jdata, jpath, NULL, false );
|
|
deleteLocalRefs( env, jname, jstrs, jbytes, jdata, jpath, DELETE_NO_REF );
|
|
}
|
|
return dict;
|
|
}
|
|
|
|
static void
|
|
and_util_getUsername( XW_UtilCtxt* uc, XWEnv xwe, XP_Bool isLocal,
|
|
XP_Bool isRobot, XP_U16 num,
|
|
XP_UCHAR* buf, XP_U16* len )
|
|
{
|
|
/* PENDING: this needs to go through to java */
|
|
XP_USE( uc );
|
|
XP_USE( xwe );
|
|
XP_USE( isRobot );
|
|
XP_USE( isLocal );
|
|
*len = XP_SNPRINTF( buf, *len-1, "Plyr #%d", num + 1 );
|
|
XP_ASSERT( '\0' == buf[*len] );
|
|
}
|
|
|
|
static void
|
|
and_util_notifyGameOver( XW_UtilCtxt* uc, XWEnv xwe, XP_S16 XP_UNUSED(quitter) )
|
|
{
|
|
UTIL_CBK_HEADER( "notifyGameOver", "()V" );
|
|
(*env)->CallVoidMethod( env, util->jutil, mid );
|
|
UTIL_CBK_TAIL();
|
|
}
|
|
|
|
#ifdef XWFEATURE_HILITECELL
|
|
static XP_Bool
|
|
and_util_hiliteCell( XW_UtilCtxt* uc, XP_U16 col, XP_U16 row )
|
|
{
|
|
/* don't log; this is getting called a lot */
|
|
return XP_TRUE; /* means keep going */
|
|
}
|
|
#endif
|
|
|
|
static XP_Bool
|
|
and_util_engineProgressCallback( XW_UtilCtxt* uc, XWEnv xwe )
|
|
{
|
|
XP_Bool result = XP_FALSE;
|
|
UTIL_CBK_HEADER("engineProgressCallback","()Z" );
|
|
result = (*env)->CallBooleanMethod( env, util->jutil, mid );
|
|
UTIL_CBK_TAIL();
|
|
return result;
|
|
}
|
|
|
|
/* This is added for java, not part of the util api */
|
|
bool
|
|
utilTimerFired( XW_UtilCtxt* uc, XWEnv xwe, XWTimerReason why, int handle )
|
|
{
|
|
bool handled = false;
|
|
AndUtil* util = (AndUtil*)uc;
|
|
TimerStorage* timerStorage = &util->timerStorage[why];
|
|
if ( handle == (int)timerStorage ) {
|
|
XWTimerProc proc = timerStorage->proc;
|
|
if ( !!proc ) {
|
|
handled = (*proc)( timerStorage->closure, xwe, why );
|
|
} else {
|
|
XP_LOGFF( "(why=%d): ERROR: no proc set", why );
|
|
}
|
|
} else {
|
|
XP_LOGFF( "mismatch: handle=%d; timerStorage=%d",
|
|
handle, (int)timerStorage );
|
|
}
|
|
return handled;
|
|
}
|
|
|
|
static void
|
|
and_util_setTimer( XW_UtilCtxt* uc, XWEnv xwe, XWTimerReason why, XP_U16 when,
|
|
XWTimerProc proc, void* closure )
|
|
{
|
|
UTIL_CBK_HEADER("setTimer", "(III)V" );
|
|
|
|
XP_ASSERT( why < VSIZE(util->timerStorage) );
|
|
TimerStorage* storage = &util->timerStorage[why];
|
|
storage->proc = proc;
|
|
storage->closure = closure;
|
|
(*env)->CallVoidMethod( env, util->jutil, mid,
|
|
why, when, (int)storage );
|
|
UTIL_CBK_TAIL();
|
|
}
|
|
|
|
static void
|
|
and_util_clearTimer( XW_UtilCtxt* uc, XWEnv xwe, XWTimerReason why )
|
|
{
|
|
UTIL_CBK_HEADER("clearTimer", "(I)V" );
|
|
(*env)->CallVoidMethod( env, util->jutil, mid, why );
|
|
UTIL_CBK_TAIL();
|
|
}
|
|
|
|
|
|
static void
|
|
and_util_requestTime( XW_UtilCtxt* uc, XWEnv xwe )
|
|
{
|
|
UTIL_CBK_HEADER("requestTime", "()V" );
|
|
(*env)->CallVoidMethod( env, util->jutil, mid );
|
|
UTIL_CBK_TAIL();
|
|
}
|
|
|
|
static XP_Bool
|
|
and_util_altKeyDown( XW_UtilCtxt* uc, XWEnv xwe )
|
|
{
|
|
LOG_FUNC();
|
|
return XP_FALSE;
|
|
}
|
|
|
|
XP_U32
|
|
and_dutil_getCurSeconds( XW_DUtilCtxt* XP_UNUSED(duc), XWEnv xwe )
|
|
{
|
|
XP_U32 curSeconds = getCurSeconds( xwe );
|
|
return curSeconds;
|
|
}
|
|
|
|
static DictionaryCtxt*
|
|
and_util_makeEmptyDict( XW_UtilCtxt* uc, XWEnv xwe )
|
|
{
|
|
#ifdef STUBBED_DICT
|
|
XP_ASSERT(0);
|
|
#else
|
|
AndGameGlobals* globals = (AndGameGlobals*)uc->closure;
|
|
DictionaryCtxt* result =
|
|
and_dictionary_make_empty( MPPARM( ((AndUtil*)uc)->util.mpool )
|
|
globals->jniutil );
|
|
return (DictionaryCtxt*)dict_ref( result, xwe );
|
|
#endif
|
|
}
|
|
|
|
static const XP_UCHAR*
|
|
and_dutil_getUserString( XW_DUtilCtxt* duc, XWEnv xwe, XP_U16 stringCode )
|
|
{
|
|
XP_UCHAR* result = "";
|
|
DUTIL_CBK_HEADER("getUserString", "(I)Ljava/lang/String;" );
|
|
int index = stringCode - 1; /* see LocalizedStrIncludes.h */
|
|
XP_ASSERT( index < VSIZE( dutil->userStrings ) );
|
|
|
|
XP_ASSERT( 0 == (dutil->userStringsBits & (1 << index)) );
|
|
|
|
if ( ! dutil->userStrings[index] ) {
|
|
jstring jresult = (*env)->CallObjectMethod( env, dutil->jdutil, mid,
|
|
stringCode );
|
|
jsize len = (*env)->GetStringUTFLength( env, jresult );
|
|
XP_UCHAR* buf = XP_MALLOC( dutil->dutil.mpool, len + 1 );
|
|
|
|
const char* jchars = (*env)->GetStringUTFChars( env, jresult, NULL );
|
|
XP_MEMCPY( buf, jchars, len );
|
|
buf[len] = '\0';
|
|
(*env)->ReleaseStringUTFChars( env, jresult, jchars );
|
|
deleteLocalRef( env, jresult );
|
|
dutil->userStrings[index] = buf;
|
|
}
|
|
|
|
result = dutil->userStrings[index];
|
|
DUTIL_CBK_TAIL();
|
|
return result;
|
|
}
|
|
|
|
static const XP_UCHAR*
|
|
and_dutil_getUserQuantityString( XW_DUtilCtxt* duc, XWEnv xwe,
|
|
XP_U16 stringCode, XP_U16 quantity )
|
|
{
|
|
XP_UCHAR* result = "";
|
|
DUTIL_CBK_HEADER("getUserQuantityString", "(II)Ljava/lang/String;" );
|
|
int index = stringCode - 1; /* see LocalizedStrIncludes.h */
|
|
XP_ASSERT( index < VSIZE( dutil->userStrings ) );
|
|
XP_UCHAR** ptrs;
|
|
|
|
dutil->userStringsBits |= 1 << index;
|
|
ptrs = (XP_UCHAR**)dutil->userStrings[index];
|
|
if ( !ptrs ) {
|
|
ptrs = (XP_UCHAR**)XP_CALLOC( dutil->dutil.mpool, MAX_QUANTITY_STRS * sizeof(*ptrs) );
|
|
dutil->userStrings[index] = (XP_UCHAR*)ptrs;
|
|
}
|
|
|
|
jstring jresult = (*env)->CallObjectMethod( env, dutil->jdutil, mid,
|
|
stringCode, quantity );
|
|
const char* jchars = (*env)->GetStringUTFChars( env, jresult, NULL );
|
|
int indx = 0;
|
|
for ( ; indx < MAX_QUANTITY_STRS; ++indx ) {
|
|
if ( !ptrs[indx] ) {
|
|
XP_LOGFF( "found empty slot %d for %s", indx, jchars );
|
|
break;
|
|
} else if ( 0 == XP_STRCMP( jchars, ptrs[indx] ) ) {
|
|
XP_LOGFF( "found %s at slot %d", jchars, indx );
|
|
break;
|
|
}
|
|
}
|
|
|
|
if ( !ptrs[indx] ) {
|
|
XP_ASSERT( indx < MAX_QUANTITY_STRS );
|
|
jsize len = (*env)->GetStringUTFLength( env, jresult );
|
|
XP_UCHAR* buf = XP_MALLOC( dutil->dutil.mpool, len + 1 );
|
|
XP_MEMCPY( buf, jchars, len );
|
|
buf[len] = '\0';
|
|
ptrs[indx] = buf;
|
|
}
|
|
|
|
(*env)->ReleaseStringUTFChars( env, jresult, jchars );
|
|
deleteLocalRef( env, jresult );
|
|
|
|
result = ptrs[indx];
|
|
DUTIL_CBK_TAIL();
|
|
return result;
|
|
}
|
|
|
|
static void
|
|
and_dutil_storePtr( XW_DUtilCtxt* duc, XWEnv xwe, const XP_UCHAR* keys[],
|
|
const void* data, XP_U32 len )
|
|
{
|
|
DUTIL_CBK_HEADER( "store", "(Ljava/lang/String;[B)V" );
|
|
XP_ASSERT( NULL == keys[1] );
|
|
|
|
jbyteArray jdata = makeByteArray( env, len, data );
|
|
jstring jkey = (*env)->NewStringUTF( env, keys[0] );
|
|
|
|
(*env)->CallVoidMethod( env, dutil->jdutil, mid, jkey, jdata );
|
|
|
|
deleteLocalRefs( env, jdata, jkey, DELETE_NO_REF );
|
|
|
|
DUTIL_CBK_TAIL();
|
|
}
|
|
|
|
static jbyteArray
|
|
loadToByteArray( XW_DUtilCtxt* duc, XWEnv xwe, const XP_UCHAR* key )
|
|
{
|
|
jbyteArray result = NULL;
|
|
DUTIL_CBK_HEADER( "load", "(Ljava/lang/String;)[B");
|
|
|
|
jstring jkey = (*env)->NewStringUTF( env, key );
|
|
result = (*env)->CallObjectMethod( env, dutil->jdutil, mid, jkey );
|
|
deleteLocalRef( env, jkey );
|
|
DUTIL_CBK_TAIL();
|
|
return result;
|
|
}
|
|
|
|
static void
|
|
and_dutil_loadPtr( XW_DUtilCtxt* duc, XWEnv xwe, const XP_UCHAR* keys[],
|
|
void* data, XP_U32* lenp )
|
|
{
|
|
XP_ASSERT( NULL == keys[1] );
|
|
JNIEnv* env = xwe;
|
|
jbyteArray jvalue = loadToByteArray( duc, env, keys[0] );
|
|
jsize len = 0;
|
|
if ( jvalue != NULL ) {
|
|
len = (*env)->GetArrayLength( env, jvalue );
|
|
if ( len <= *lenp ) {
|
|
jbyte* jelems = (*env)->GetByteArrayElements( env, jvalue, NULL );
|
|
XP_MEMCPY( data, jelems, len );
|
|
(*env)->ReleaseByteArrayElements( env, jvalue, jelems, 0 );
|
|
}
|
|
deleteLocalRef( env, jvalue );
|
|
}
|
|
*lenp = len;
|
|
}
|
|
|
|
#ifdef XWFEATURE_DEVICE
|
|
static void
|
|
and_dutil_forEach( XW_DUtilCtxt* duc, XWEnv xwe, const XP_UCHAR* keys[],
|
|
OnOneProc proc, void* closure )
|
|
{
|
|
XP_ASSERT(0);
|
|
}
|
|
|
|
static void
|
|
and_dutil_remove( XW_DUtilCtxt* duc, const XP_UCHAR* keys[] )
|
|
{
|
|
XP_ASSERT(0);
|
|
}
|
|
#endif
|
|
|
|
static void
|
|
and_util_notifyIllegalWords( XW_UtilCtxt* uc, XWEnv xwe, BadWordInfo* bwi,
|
|
XP_U16 turn, XP_Bool turnLost )
|
|
{
|
|
UTIL_CBK_HEADER("notifyIllegalWords",
|
|
"(Ljava/lang/String;[Ljava/lang/String;IZ)V" );
|
|
XP_ASSERT( bwi->nWords > 0 );
|
|
if ( bwi->nWords > 0 ) {
|
|
jobjectArray jwords = makeStringArray( env, bwi->nWords,
|
|
(const XP_UCHAR**)bwi->words );
|
|
XP_ASSERT( !!bwi->dictName );
|
|
jstring jname = (*env)->NewStringUTF( env, bwi->dictName );
|
|
(*env)->CallVoidMethod( env, util->jutil, mid,
|
|
jname, jwords, turn, turnLost );
|
|
deleteLocalRefs( env, jwords, jname, DELETE_NO_REF );
|
|
}
|
|
UTIL_CBK_TAIL();
|
|
}
|
|
|
|
#ifdef XWFEATURE_CHAT
|
|
static void
|
|
and_util_showChat( XW_UtilCtxt* uc, XWEnv xwe, const XP_UCHAR* msg, XP_S16 from, XP_U32 timestamp )
|
|
{
|
|
UTIL_CBK_HEADER( "showChat", "(Ljava/lang/String;ILjava/lang/String;I)V" );
|
|
jstring jname = NULL;
|
|
if ( 0 <= from ) {
|
|
LocalPlayer* lp = &uc->gameInfo->players[from];
|
|
XP_ASSERT( !lp->isLocal );
|
|
jname = (*env)->NewStringUTF( env, lp->name );
|
|
}
|
|
|
|
jstring jmsg = (*env)->NewStringUTF( env, msg );
|
|
(*env)->CallVoidMethod( env, util->jutil, mid, jmsg, from, jname, timestamp );
|
|
deleteLocalRefs( env, jmsg, jname, DELETE_NO_REF );
|
|
UTIL_CBK_TAIL();
|
|
}
|
|
#endif
|
|
|
|
static void
|
|
and_util_remSelected( XW_UtilCtxt* uc, XWEnv xwe )
|
|
{
|
|
UTIL_CBK_HEADER("remSelected", "()V" );
|
|
(*env)->CallVoidMethod( env, util->jutil, mid );
|
|
UTIL_CBK_TAIL();
|
|
}
|
|
|
|
static void
|
|
and_util_getMQTTIDsFor( XW_UtilCtxt* uc, XWEnv xwe, XP_U16 nRelayIDs,
|
|
const XP_UCHAR* relayIDs[] )
|
|
{
|
|
UTIL_CBK_HEADER("getMQTTIDsFor", "([Ljava/lang/String;)V" );
|
|
|
|
jobjectArray jrids = makeStringArray( env, nRelayIDs, relayIDs );
|
|
(*env)->CallVoidMethod( env, util->jutil, mid, jrids );
|
|
deleteLocalRef( env, jrids );
|
|
UTIL_CBK_TAIL();
|
|
}
|
|
|
|
static void
|
|
and_util_timerSelected( XW_UtilCtxt* uc, XWEnv xwe, XP_Bool inDuplicateMode, XP_Bool canPause )
|
|
{
|
|
UTIL_CBK_HEADER("timerSelected", "(ZZ)V" );
|
|
(*env)->CallVoidMethod( env, util->jutil, mid, inDuplicateMode, canPause );
|
|
UTIL_CBK_TAIL();
|
|
}
|
|
|
|
static void
|
|
and_util_formatPauseHistory( XW_UtilCtxt* uc, XWEnv xwe, XWStreamCtxt* stream,
|
|
DupPauseType typ, XP_S16 turn,
|
|
XP_U32 secsPrev, XP_U32 secsCur,
|
|
const XP_UCHAR* msg )
|
|
{
|
|
UTIL_CBK_HEADER( "formatPauseHistory",
|
|
"(IIIILjava/lang/String;)Ljava/lang/String;" );
|
|
jstring jmsg = !! msg ? (*env)->NewStringUTF( env, msg ) : NULL;
|
|
|
|
jstring jresult = (*env)->CallObjectMethod( env, util->jutil, mid, typ,
|
|
turn, secsPrev, secsCur, jmsg );
|
|
|
|
const char* jchars = (*env)->GetStringUTFChars( env, jresult, NULL );
|
|
stream_catString( stream, jchars );
|
|
(*env)->ReleaseStringUTFChars( env, jresult, jchars );
|
|
deleteLocalRefs( env, jresult, jmsg, DELETE_NO_REF );
|
|
UTIL_CBK_TAIL();
|
|
}
|
|
|
|
#ifndef XWFEATURE_MINIWIN
|
|
static void
|
|
and_util_bonusSquareHeld( XW_UtilCtxt* uc, XWEnv xwe, XWBonusType bonus )
|
|
{
|
|
UTIL_CBK_HEADER( "bonusSquareHeld", "(I)V" );
|
|
(*env)->CallVoidMethod( env, util->jutil, mid, bonus );
|
|
UTIL_CBK_TAIL();
|
|
}
|
|
|
|
static void
|
|
and_util_playerScoreHeld( XW_UtilCtxt* uc, XWEnv xwe, XP_U16 player )
|
|
{
|
|
UTIL_CBK_HEADER( "playerScoreHeld", "(I)V" );
|
|
(*env)->CallVoidMethod( env, util->jutil, mid, player );
|
|
UTIL_CBK_TAIL();
|
|
}
|
|
#endif
|
|
|
|
#ifdef XWFEATURE_SMS
|
|
static XP_Bool
|
|
and_dutil_phoneNumbersSame( XW_DUtilCtxt* duc, XWEnv xwe, const XP_UCHAR* p1,
|
|
const XP_UCHAR* p2 )
|
|
{
|
|
XP_Bool same = 0 == strcmp( p1, p2 );
|
|
if ( !same ) {
|
|
DUTIL_CBK_HEADER( "phoneNumbersSame",
|
|
"(Ljava/lang/String;Ljava/lang/String;)Z" );
|
|
jstring js1 = (*env)->NewStringUTF( env, p1 );
|
|
jstring js2 = (*env)->NewStringUTF( env, p2 );
|
|
same = (*env)->CallBooleanMethod( env, dutil->jdutil, mid, js1, js2 );
|
|
deleteLocalRefs( env, js1, js2, DELETE_NO_REF );
|
|
DUTIL_CBK_TAIL();
|
|
}
|
|
return same;
|
|
}
|
|
#endif
|
|
|
|
#ifdef XWFEATURE_BOARDWORDS
|
|
static void
|
|
and_util_cellSquareHeld( XW_UtilCtxt* uc, XWEnv xwe, XWStreamCtxt* words )
|
|
{
|
|
if ( NULL != words ) {
|
|
UTIL_CBK_HEADER( "cellSquareHeld", "(Ljava/lang/String;)V" );
|
|
jstring jwords = streamToJString( env, words );
|
|
(*env)->CallVoidMethod( env, util->jutil, mid, jwords );
|
|
deleteLocalRef( env, jwords );
|
|
UTIL_CBK_TAIL();
|
|
}
|
|
}
|
|
#endif
|
|
|
|
#ifndef XWFEATURE_STANDALONE_ONLY
|
|
|
|
static void
|
|
and_util_informMissing( XW_UtilCtxt* uc, XWEnv xwe, XP_Bool isServer,
|
|
const CommsAddrRec* addr, XP_U16 nDevs, XP_U16 nMissing)
|
|
{
|
|
UTIL_CBK_HEADER( "informMissing",
|
|
"(ZL" PKG_PATH("jni/CommsAddrRec$CommsConnTypeSet") ";II)V" );
|
|
jobject jtypset = NULL;
|
|
if ( !!addr ) {
|
|
jtypset = addrTypesToJ( env, addr );
|
|
}
|
|
(*env)->CallVoidMethod( env, util->jutil, mid, isServer, jtypset, nDevs,
|
|
nMissing );
|
|
deleteLocalRef( env, jtypset );
|
|
UTIL_CBK_TAIL();
|
|
}
|
|
|
|
static void
|
|
and_util_addrChange( XW_UtilCtxt* uc, XWEnv xwe,
|
|
const CommsAddrRec* oldAddr,
|
|
const CommsAddrRec* newAddr )
|
|
{
|
|
// LOG_FUNC();
|
|
}
|
|
|
|
static void
|
|
and_util_informWordsBlocked( XW_UtilCtxt* uc, XWEnv xwe, XP_U16 nBadWords,
|
|
XWStreamCtxt* words, const XP_UCHAR* dict )
|
|
{
|
|
UTIL_CBK_HEADER( "informWordsBlocked", "(ILjava/lang/String;Ljava/lang/String;)V" );
|
|
jstring jwords = streamToJString( env, words );
|
|
jstring jdict = (*env)->NewStringUTF( env, dict );
|
|
(*env)->CallVoidMethod( env, util->jutil, mid, nBadWords, jwords, jdict );
|
|
deleteLocalRefs( env, jwords, jdict, DELETE_NO_REF );
|
|
UTIL_CBK_TAIL();
|
|
}
|
|
|
|
static void
|
|
and_util_getInviteeName( XW_UtilCtxt* uc, XWEnv xwe, XP_U16 plyrNum,
|
|
XP_UCHAR* buf, XP_U16* bufLen )
|
|
{
|
|
UTIL_CBK_HEADER( "getInviteeName", "(I)Ljava/lang/String;" );
|
|
jstring jresult = (*env)->CallObjectMethod( env, util->jutil, mid, plyrNum );
|
|
if ( NULL != jresult ) {
|
|
jsize len = (*env)->GetStringUTFLength( env, jresult );
|
|
if ( len < *bufLen ) {
|
|
const char* jchars = (*env)->GetStringUTFChars( env, jresult, NULL );
|
|
XP_STRCAT( buf, jchars );
|
|
(*env)->ReleaseStringUTFChars( env, jresult, jchars );
|
|
*bufLen = len;
|
|
} else {
|
|
*bufLen = 0;
|
|
}
|
|
deleteLocalRef( env, jresult );
|
|
}
|
|
UTIL_CBK_TAIL();
|
|
}
|
|
|
|
#if defined XWFEATURE_DEVID && defined XWFEATURE_RELAY
|
|
static const XP_UCHAR*
|
|
and_dutil_getDevID( XW_DUtilCtxt* duc, XWEnv xwe, DevIDType* typ )
|
|
{
|
|
const XP_UCHAR* result = NULL;
|
|
*typ = ID_TYPE_NONE;
|
|
DUTIL_CBK_HEADER( "getDevID", "([B)Ljava/lang/String;" );
|
|
jbyteArray jbarr = makeByteArray( env, 1, NULL );
|
|
jstring jresult = (*env)->CallObjectMethod( env, dutil->jdutil, mid, jbarr );
|
|
if ( NULL != jresult ) {
|
|
const char* jchars = (*env)->GetStringUTFChars( env, jresult, NULL );
|
|
jsize len = (*env)->GetStringUTFLength( env, jresult );
|
|
if ( NULL != dutil->devIDStorage
|
|
&& 0 == XP_MEMCMP( dutil->devIDStorage, jchars, len ) ) {
|
|
XP_LOGFF( "already have matching devID" );
|
|
} else {
|
|
XP_LOGFF( "allocating storage for devID" );
|
|
XP_FREEP( dutil->dutil.mpool, &dutil->devIDStorage );
|
|
dutil->devIDStorage = XP_MALLOC( dutil->dutil.mpool, len + 1 );
|
|
XP_MEMCPY( dutil->devIDStorage, jchars, len );
|
|
dutil->devIDStorage[len] = '\0';
|
|
}
|
|
(*env)->ReleaseStringUTFChars( env, jresult, jchars );
|
|
result = (const XP_UCHAR*)dutil->devIDStorage;
|
|
|
|
jbyte* elems = (*env)->GetByteArrayElements( env, jbarr, NULL );
|
|
*typ = (DevIDType)elems[0];
|
|
(*env)->ReleaseByteArrayElements( env, jbarr, elems, 0 );
|
|
}
|
|
deleteLocalRefs( env, jbarr, jresult, DELETE_NO_REF );
|
|
DUTIL_CBK_TAIL();
|
|
return result;
|
|
}
|
|
|
|
static void
|
|
and_dutil_deviceRegistered( XW_DUtilCtxt* duc, XWEnv xwe, DevIDType typ,
|
|
const XP_UCHAR* idRelay )
|
|
{
|
|
DUTIL_CBK_HEADER( "deviceRegistered",
|
|
"(L" PKG_PATH("jni/DUtilCtxt$DevIDType") ";Ljava/lang/String;)V" );
|
|
jstring jstr = (*env)->NewStringUTF( env, idRelay );
|
|
jobject jtyp = intToJEnum( env, typ,
|
|
PKG_PATH("jni/DUtilCtxt$DevIDType") );
|
|
(*env)->CallVoidMethod( env, dutil->jdutil, mid, jtyp, jstr );
|
|
deleteLocalRefs( env, jstr, jtyp, DELETE_NO_REF );
|
|
DUTIL_CBK_TAIL();
|
|
}
|
|
#endif /* XWFEATURE_DEVID && XWFEATURE_RELAY */
|
|
|
|
#endif
|
|
|
|
#ifdef XWFEATURE_SEARCHLIMIT
|
|
static XP_Bool
|
|
and_util_getTraySearchLimits(XW_UtilCtxt* uc, XP_U16* min, XP_U16* max )
|
|
{
|
|
LOG_FUNC();
|
|
foobar; /* this should not be compiling */
|
|
}
|
|
|
|
#endif
|
|
|
|
#ifdef SHOW_PROGRESS
|
|
static void
|
|
and_util_engineStarting( XW_UtilCtxt* uc, XP_U16 nBlanks )
|
|
{
|
|
UTIL_CBK_HEADER("engineStarting", "(I)V" );
|
|
(*env)->CallVoidMethod( env, util->jutil, mid, nBlanks );
|
|
UTIL_CBK_TAIL();
|
|
}
|
|
|
|
static void
|
|
and_util_engineStopping( XW_UtilCtxt* uc )
|
|
{
|
|
UTIL_CBK_HEADER("engineStopping", "()V" );
|
|
(*env)->CallVoidMethod( env, util->jutil, mid );
|
|
UTIL_CBK_TAIL();
|
|
}
|
|
#endif
|
|
|
|
static XW_DUtilCtxt*
|
|
and_util_getDevUtilCtxt( XW_UtilCtxt* uc, XWEnv xwe )
|
|
{
|
|
AndGameGlobals* globals = (AndGameGlobals*)uc->closure;
|
|
XP_ASSERT( !!globals->dutil );
|
|
return globals->dutil;
|
|
}
|
|
|
|
#ifdef COMMS_CHECKSUM
|
|
static XP_UCHAR*
|
|
and_dutil_md5sum( XW_DUtilCtxt* duc, XWEnv xwe, const XP_U8* ptr, XP_U32 len )
|
|
{
|
|
/* will be signed in the java world, and negative is bad! */
|
|
XP_ASSERT( (0x80000000 & len) == 0 );
|
|
AndDUtil* dutil = (AndDUtil*)duc;
|
|
JNIEnv* env = xwe;
|
|
struct JNIUtilCtxt* jniutil = dutil->jniutil;
|
|
jstring jsum = and_util_getMD5SumForBytes( jniutil, env, ptr, len );
|
|
XP_UCHAR* result = getStringCopy( MPPARM(duc->mpool) env, jsum );
|
|
deleteLocalRef( env, jsum );
|
|
return result;
|
|
}
|
|
#endif
|
|
|
|
static void
|
|
and_dutil_notifyPause( XW_DUtilCtxt* duc, XWEnv xwe, XP_U32 gameID, DupPauseType pauseTyp,
|
|
XP_U16 pauser, const XP_UCHAR* name,
|
|
const XP_UCHAR* msg )
|
|
{
|
|
DUTIL_CBK_HEADER( "notifyPause", "(IIILjava/lang/String;Ljava/lang/String;)V" );
|
|
jstring jname = (*env)->NewStringUTF( env, name );
|
|
jstring jmsg = (*env)->NewStringUTF( env, msg );
|
|
(*env)->CallVoidMethod( env, dutil->jdutil, mid, gameID, pauseTyp, pauser,
|
|
jname, jmsg );
|
|
deleteLocalRefs( env, jname, jmsg, DELETE_NO_REF );
|
|
DUTIL_CBK_TAIL();
|
|
}
|
|
|
|
static XP_Bool
|
|
and_dutil_haveGame( XW_DUtilCtxt* duc, XWEnv xwe, XP_U32 gameID, XP_U8 channel )
|
|
{
|
|
XP_Bool result;
|
|
DUTIL_CBK_HEADER( "haveGame", "(II)Z" );
|
|
result = (*env)->CallBooleanMethod( env, dutil->jdutil, mid, gameID, channel );
|
|
DUTIL_CBK_TAIL();
|
|
return result;
|
|
}
|
|
|
|
static void
|
|
and_dutil_onDupTimerChanged( XW_DUtilCtxt* duc, XWEnv xwe, XP_U32 gameID,
|
|
XP_U32 oldVal, XP_U32 newVal )
|
|
{
|
|
DUTIL_CBK_HEADER( "onDupTimerChanged", "(III)V" );
|
|
(*env)->CallVoidMethod( env, dutil->jdutil, mid, gameID, oldVal, newVal );
|
|
DUTIL_CBK_TAIL();
|
|
}
|
|
|
|
static void
|
|
and_dutil_onInviteReceived( XW_DUtilCtxt* duc, XWEnv xwe, const NetLaunchInfo* nli )
|
|
{
|
|
LOGNLI( nli );
|
|
DUTIL_CBK_HEADER( "onInviteReceived", "(L" PKG_PATH("NetLaunchInfo") ";)V" );
|
|
|
|
/* Allocate a new NetLaunchInfo */
|
|
jobject jnli = makeObjectEmptyConst( env, PKG_PATH("NetLaunchInfo") );
|
|
XP_ASSERT( !!jnli );
|
|
setNLI( env, jnli, nli );
|
|
|
|
(*env)->CallVoidMethod( env, dutil->jdutil, mid, jnli );
|
|
|
|
deleteLocalRef( env, jnli );
|
|
DUTIL_CBK_TAIL();
|
|
}
|
|
|
|
static void
|
|
and_dutil_onMessageReceived( XW_DUtilCtxt* duc, XWEnv xwe, XP_U32 gameID,
|
|
const CommsAddrRec* from, const XP_U8* data, XP_U16 len )
|
|
{
|
|
LOG_FUNC();
|
|
DUTIL_CBK_HEADER( "onMessageReceived",
|
|
"(IL" PKG_PATH("jni/CommsAddrRec") ";[B)V" );
|
|
|
|
jbyteArray jmsg = makeByteArray( env, len, (jbyte*)data );
|
|
|
|
jobject jaddr = makeJAddr( env, from );
|
|
|
|
(*env)->CallVoidMethod( env, dutil->jdutil, mid, gameID, jaddr, jmsg );
|
|
|
|
deleteLocalRefs( env, jmsg, jaddr, DELETE_NO_REF );
|
|
|
|
DUTIL_CBK_TAIL();
|
|
LOG_RETURN_VOID();
|
|
}
|
|
|
|
static void
|
|
and_dutil_onGameGoneReceived( XW_DUtilCtxt* duc, XWEnv xwe, XP_U32 gameID,
|
|
const CommsAddrRec* from )
|
|
{
|
|
DUTIL_CBK_HEADER( "onGameGoneReceived",
|
|
"(IL" PKG_PATH("jni/CommsAddrRec") ";)V" );
|
|
jobject jaddr = makeJAddr( env, from );
|
|
(*env)->CallVoidMethod( env, dutil->jdutil, mid, gameID, jaddr );
|
|
|
|
deleteLocalRefs( env, jaddr, DELETE_NO_REF );
|
|
DUTIL_CBK_TAIL();
|
|
}
|
|
|
|
static void
|
|
and_dutil_ackMQTTMsg( XW_DUtilCtxt* duc, XWEnv xwe, XP_U32 gameID,
|
|
const MQTTDevID* senderID, const XP_U8* msg, XP_U16 len )
|
|
{
|
|
DUTIL_CBK_HEADER( "ackMQTTMsg", "(ILjava/lang/String;[B)V" );
|
|
|
|
XP_UCHAR tmp[32];
|
|
formatMQTTDevID( senderID, tmp, VSIZE(tmp) );
|
|
jstring jdevid = (*env)->NewStringUTF( env, tmp );
|
|
jbyteArray jmsg = makeByteArray( env, len, (const jbyte*)msg );
|
|
|
|
(*env)->CallVoidMethod( env, dutil->jdutil, mid, gameID, jdevid, jmsg );
|
|
|
|
deleteLocalRefs( env, jdevid, jmsg, DELETE_NO_REF );
|
|
DUTIL_CBK_TAIL();
|
|
}
|
|
|
|
XW_UtilCtxt*
|
|
makeUtil( MPFORMAL JNIEnv* env,
|
|
#ifdef MAP_THREAD_TO_ENV
|
|
EnvThreadInfo* ti,
|
|
#endif
|
|
jobject jutil, CurGameInfo* gi,
|
|
AndGameGlobals* closure )
|
|
{
|
|
AndUtil* util = (AndUtil*)XP_CALLOC( mpool, sizeof(*util) );
|
|
#ifdef MAP_THREAD_TO_ENV
|
|
util->ti = ti;
|
|
#endif
|
|
UtilVtable* vtable = (UtilVtable*)XP_CALLOC( mpool, sizeof(*vtable) );
|
|
if ( NULL != jutil ) {
|
|
util->jutil = (*env)->NewGlobalRef( env, jutil );
|
|
}
|
|
util->util.vtable = vtable;
|
|
MPASSIGN( util->util.mpool, mpool );
|
|
util->util.closure = closure;
|
|
util->util.gameInfo = gi;
|
|
|
|
#define SET_PROC(nam) vtable->m_util_##nam = and_util_##nam
|
|
|
|
#ifndef XWFEATURE_STANDALONE_ONLY
|
|
SET_PROC(makeStreamFromAddr);
|
|
#endif
|
|
SET_PROC(userError);
|
|
SET_PROC(notifyMove);
|
|
SET_PROC(notifyTrade);
|
|
SET_PROC(notifyPickTileBlank);
|
|
SET_PROC(informNeedPickTiles);
|
|
SET_PROC(informNeedPassword);
|
|
SET_PROC(trayHiddenChange);
|
|
SET_PROC(yOffsetChange);
|
|
#ifdef XWFEATURE_TURNCHANGENOTIFY
|
|
SET_PROC(turnChanged);
|
|
#endif
|
|
SET_PROC(informMove);
|
|
SET_PROC(notifyDupStatus);
|
|
SET_PROC(informUndo);
|
|
SET_PROC(informNetDict);
|
|
SET_PROC(getDict);
|
|
SET_PROC(getUsername);
|
|
SET_PROC(notifyGameOver);
|
|
#ifdef XWFEATURE_HILITECELL
|
|
SET_PROC(hiliteCell);
|
|
#endif
|
|
SET_PROC(engineProgressCallback);
|
|
SET_PROC(setTimer);
|
|
SET_PROC(clearTimer);
|
|
SET_PROC(requestTime);
|
|
SET_PROC(altKeyDown);
|
|
SET_PROC(makeEmptyDict);
|
|
SET_PROC(notifyIllegalWords);
|
|
#ifdef XWFEATURE_CHAT
|
|
SET_PROC(showChat);
|
|
#endif
|
|
SET_PROC(remSelected);
|
|
SET_PROC(getMQTTIDsFor);
|
|
SET_PROC(timerSelected);
|
|
SET_PROC(formatPauseHistory);
|
|
|
|
#ifndef XWFEATURE_MINIWIN
|
|
SET_PROC(bonusSquareHeld);
|
|
SET_PROC(playerScoreHeld);
|
|
#endif
|
|
|
|
#ifdef XWFEATURE_BOARDWORDS
|
|
SET_PROC(cellSquareHeld);
|
|
#endif
|
|
|
|
#ifndef XWFEATURE_STANDALONE_ONLY
|
|
SET_PROC(informMissing);
|
|
SET_PROC(addrChange);
|
|
#endif
|
|
#ifdef XWFEATURE_SEARCHLIMIT
|
|
SET_PROC(getTraySearchLimits);
|
|
#endif
|
|
#ifdef SHOW_PROGRESS
|
|
SET_PROC(engineStarting);
|
|
SET_PROC(engineStopping);
|
|
#endif
|
|
|
|
SET_PROC(getDevUtilCtxt);
|
|
SET_PROC(informWordsBlocked);
|
|
SET_PROC(getInviteeName);
|
|
|
|
#undef SET_PROC
|
|
assertTableFull( vtable, sizeof(*vtable), "util" );
|
|
return (XW_UtilCtxt*)util;
|
|
} /* makeUtil */
|
|
|
|
void
|
|
destroyUtil( XW_UtilCtxt** utilc, JNIEnv* env )
|
|
{
|
|
AndUtil* util = (AndUtil*)*utilc;
|
|
|
|
if ( NULL != util->jutil ) {
|
|
(*env)->DeleteGlobalRef( env, util->jutil );
|
|
}
|
|
XP_FREE( util->util.mpool, util->util.vtable );
|
|
XP_FREE( util->util.mpool, util );
|
|
*utilc = NULL;
|
|
}
|
|
|
|
XW_DUtilCtxt*
|
|
makeDUtil( MPFORMAL JNIEnv* env,
|
|
#ifdef MAP_THREAD_TO_ENV
|
|
EnvThreadInfo* ti,
|
|
#endif
|
|
jobject jdutil, VTableMgr* vtMgr,
|
|
DictMgrCtxt* dmgr, JNIUtilCtxt* jniutil,
|
|
void* closure )
|
|
{
|
|
AndDUtil* dutil = (AndDUtil*)XP_CALLOC( mpool, sizeof(*dutil) );
|
|
dutil_super_init( MPPARM(mpool) &dutil->dutil );
|
|
#ifdef MAP_THREAD_TO_ENV
|
|
dutil->ti = ti;
|
|
#endif
|
|
dutil->jniutil = jniutil;
|
|
dutil->dutil.closure = closure;
|
|
dutil->dutil.vtMgr = vtMgr;
|
|
dutil->dictMgr = dmgr;
|
|
|
|
if ( NULL != jdutil ) {
|
|
dutil->jdutil = (*env)->NewGlobalRef( env, jdutil );
|
|
}
|
|
|
|
DUtilVtable* vtable = &dutil->dutil.vtable;
|
|
#define SET_DPROC(nam) vtable->m_dutil_##nam = and_dutil_##nam
|
|
SET_DPROC(getCurSeconds);
|
|
SET_DPROC(getUserString);
|
|
SET_DPROC(getUserQuantityString);
|
|
SET_DPROC(storePtr);
|
|
SET_DPROC(loadPtr);
|
|
# ifdef XWFEATURE_DEVICE
|
|
SET_DPROC(forEach);
|
|
SET_DPROC(remove);
|
|
# endif
|
|
|
|
# if defined XWFEATURE_DEVID && defined XWFEATURE_RELAY
|
|
SET_DPROC(getDevID);
|
|
SET_DPROC(deviceRegistered);
|
|
# endif
|
|
#ifdef XWFEATURE_SMS
|
|
SET_DPROC(phoneNumbersSame);
|
|
#endif
|
|
#ifdef COMMS_CHECKSUM
|
|
SET_DPROC(md5sum);
|
|
#endif
|
|
SET_DPROC(notifyPause);
|
|
SET_DPROC(haveGame);
|
|
SET_DPROC(onDupTimerChanged);
|
|
|
|
SET_DPROC(onInviteReceived);
|
|
SET_DPROC(onMessageReceived);
|
|
SET_DPROC(onGameGoneReceived);
|
|
SET_DPROC(ackMQTTMsg);
|
|
|
|
#undef SET_DPROC
|
|
|
|
assertTableFull( vtable, sizeof(*vtable), "dutil" );
|
|
return &dutil->dutil;
|
|
}
|
|
|
|
void
|
|
destroyDUtil( XW_DUtilCtxt** dutilp, JNIEnv* env )
|
|
{
|
|
AndDUtil* dutil = (AndDUtil*)*dutilp;
|
|
if ( NULL != dutil->jdutil ) {
|
|
(*env)->DeleteGlobalRef( env, dutil->jdutil );
|
|
}
|
|
|
|
for ( int ii = 0; ii < VSIZE(dutil->userStrings); ++ii ) {
|
|
XP_UCHAR* ptr = dutil->userStrings[ii];
|
|
if ( NULL != ptr ) {
|
|
if ( 0 == (dutil->userStringsBits & (1 << ii)) ) {
|
|
XP_FREE( dutil->dutil.mpool, ptr );
|
|
} else {
|
|
XP_UCHAR** ptrs = (XP_UCHAR**)ptr;
|
|
for ( int jj = 0; jj < MAX_QUANTITY_STRS; ++jj ) {
|
|
ptr = ptrs[jj];
|
|
if ( !!ptr ) {
|
|
XP_FREE( dutil->dutil.mpool, ptr );
|
|
}
|
|
}
|
|
XP_FREE( dutil->dutil.mpool, ptrs );
|
|
}
|
|
}
|
|
}
|
|
#ifdef XWFEATURE_DEVID
|
|
XP_FREEP( dutil->dutil.mpool, &dutil->devIDStorage );
|
|
#endif
|
|
}
|