/* -*- compile-command: "find-and-gradle.sh inXw4dDeb"; -*- */ /* * Copyright 2001 - 2023 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 #include #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 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, xwe ); 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( ((AndDUtil*)duc)->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_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 static void and_util_informMissing( XW_UtilCtxt* uc, XWEnv xwe, XP_Bool isServer, const CommsAddrRec* hostAddr, const CommsAddrRec* selfAddr, XP_U16 nDevs, XP_U16 nMissing, XP_U16 nInvited, XP_Bool fromRematch ) { UTIL_CBK_HEADER( "informMissing", "(ZL" PKG_PATH("jni/CommsAddrRec") ";" "L" PKG_PATH("jni/CommsAddrRec$CommsConnTypeSet") ";IIIZ)V" ); jobject jHostAddr = NULL; if ( !!hostAddr ) { jHostAddr = makeJAddr( env, hostAddr ); } jobject jtypset = NULL; if ( !!selfAddr ) { jtypset = addrTypesToJ( env, selfAddr ); } (*env)->CallVoidMethod( env, util->jutil, mid, isServer, jHostAddr, jtypset, nDevs, nMissing, nInvited, fromRematch ); deleteLocalRefs( env, jHostAddr, jtypset, DELETE_NO_REF ); UTIL_CBK_TAIL(); } 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 */ #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_getUsername( XW_DUtilCtxt* duc, XWEnv xwe, XP_U16 num, XP_Bool isLocal, XP_Bool isRobot, XP_UCHAR* buf, XP_U16* lenp ) { DUTIL_CBK_HEADER( "getUsername", "(IZZ)Ljava/lang/String;" ); jstring jresult = (*env)->CallObjectMethod( env, dutil->jdutil, mid, num, isLocal, isRobot ); jsize len = (*env)->GetStringUTFLength( env, jresult ); if ( len < *lenp ) { const char* jchars = (*env)->GetStringUTFChars( env, jresult, NULL ); *lenp = XP_SNPRINTF( buf, len+1, "%s", jchars ); (*env)->ReleaseStringUTFChars( env, jresult, jchars ); deleteLocalRef( env, jresult ); } DUTIL_CBK_TAIL(); } 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 = XP_FALSE; DUTIL_CBK_HEADER( "haveGame", "(II)Z" ); XP_ASSERT( !!dutil->jdutil ); /* returning false by default a problem */ 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 = makeObjectEmptyConstr( 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_onCtrlReceived( XW_DUtilCtxt* duc, XWEnv xwe, const XP_U8* buf, XP_U16 len ) { DUTIL_CBK_HEADER( "onCtrlReceived", "([B)V" ); jbyteArray jmsg = makeByteArray( env, len, (jbyte*)buf ); (*env)->CallVoidMethod( env, dutil->jdutil, mid, jmsg ); deleteLocalRef( env, jmsg ); DUTIL_CBK_TAIL(); } 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, const XP_UCHAR* topic, XP_U32 gameID, const MQTTDevID* senderID, const XP_U8* msg, XP_U16 len ) { DUTIL_CBK_HEADER( "ackMQTTMsg", "(Ljava/lang/String;ILjava/lang/String;[B)V" ); XP_UCHAR tmp[32]; formatMQTTDevID( senderID, tmp, VSIZE(tmp) ); jstring jTopic = (*env)->NewStringUTF( env, topic ); jstring jdevid = (*env)->NewStringUTF( env, tmp ); jbyteArray jmsg = makeByteArray( env, len, (const jbyte*)msg ); (*env)->CallVoidMethod( env, dutil->jdutil, mid, jTopic, 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(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 SET_PROC(informMissing); #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(getUsername); SET_DPROC(notifyPause); SET_DPROC(haveGame); SET_DPROC(onDupTimerChanged); SET_DPROC(onInviteReceived); SET_DPROC(onMessageReceived); SET_DPROC(onCtrlReceived); 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 }