mirror of
git://xwords.git.sourceforge.net/gitroot/xwords/xwords
synced 2025-01-10 05:26:10 +01:00
e8ab9e0b8b
My VSIZE is no longer legal, and apparently there's no workaround (no way to safely figure the length of an array whose size is known at compile time.) To avoid the risk of duplicating little constants, added macros that define length in a way a new VSIZE can pick it up. Couldn't make that work for struct field arrays, however, so there I'm using constants.
677 lines
21 KiB
C
677 lines
21 KiB
C
/* -*- compile-command: "find-and-ant.sh debug install"; -*- */
|
|
/*
|
|
* Copyright © 2009 - 2016 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 <jni.h>
|
|
#include <stdio.h>
|
|
#include <stdlib.h>
|
|
#include <sys/types.h>
|
|
#include <sys/stat.h>
|
|
#include <unistd.h>
|
|
#include <fcntl.h>
|
|
#include <sys/mman.h>
|
|
|
|
#include "anddict.h"
|
|
#include "xptypes.h"
|
|
#include "dictnry.h"
|
|
#include "dictnryp.h"
|
|
#include "strutils.h"
|
|
#include "andutils.h"
|
|
#include "utilwrapper.h"
|
|
|
|
typedef struct _AndDictionaryCtxt {
|
|
DictionaryCtxt super;
|
|
JNIUtilCtxt* jniutil;
|
|
JNIEnv *env;
|
|
off_t bytesSize;
|
|
jbyte* bytes;
|
|
jbyteArray byteArray;
|
|
#ifdef DEBUG
|
|
uint32_t dbgid;
|
|
#endif
|
|
} AndDictionaryCtxt;
|
|
|
|
#define CHECK_PTR(p,c,e) \
|
|
if ( ((p)+(c)) > (e) ) { \
|
|
XP_LOGF( "%s (line %d); out of bytes", __func__, __LINE__ ); \
|
|
goto error; \
|
|
}
|
|
|
|
static void splitFaces_via_java( JNIEnv* env, AndDictionaryCtxt* ctxt,
|
|
const XP_U8* ptr,
|
|
int nFaceBytes, int nFaces, XP_Bool isUTF8 );
|
|
|
|
void
|
|
dict_splitFaces( DictionaryCtxt* dict, const XP_U8* bytes,
|
|
XP_U16 nBytes, XP_U16 nFaces )
|
|
{
|
|
AndDictionaryCtxt* ctxt = (AndDictionaryCtxt*)dict;
|
|
splitFaces_via_java( ctxt->env, ctxt, bytes, nBytes, nFaces,
|
|
dict->isUTF8 );
|
|
}
|
|
|
|
static XP_U32
|
|
n_ptr_tohl( XP_U8 const** inp )
|
|
{
|
|
XP_U32 t;
|
|
XP_MEMCPY( &t, *inp, sizeof(t) );
|
|
|
|
*inp += sizeof(t);
|
|
|
|
return XP_NTOHL(t);
|
|
} /* n_ptr_tohl */
|
|
|
|
static XP_U16
|
|
n_ptr_tohs( XP_U8 const ** inp )
|
|
{
|
|
XP_U16 t;
|
|
XP_MEMCPY( &t, *inp, sizeof(t) );
|
|
|
|
*inp += sizeof(t);
|
|
|
|
return XP_NTOHS(t);
|
|
} /* n_ptr_tohs */
|
|
|
|
static XP_U16
|
|
andCountSpecials( AndDictionaryCtxt* ctxt )
|
|
{
|
|
XP_U16 result = 0;
|
|
|
|
for ( int ii = 0; ii < ctxt->super.nFaces; ++ii ) {
|
|
if ( IS_SPECIAL( ctxt->super.facePtrs[ii][0] ) ) {
|
|
++result;
|
|
}
|
|
}
|
|
|
|
return result;
|
|
} /* andCountSpecials */
|
|
|
|
static XP_Bool
|
|
andMakeBitmap( AndDictionaryCtxt* ctxt, XP_U8 const** ptrp,
|
|
const XP_U8* end, XP_Bitmap* result )
|
|
{
|
|
XP_Bool success = XP_TRUE;
|
|
XP_U8 const* ptr = *ptrp;
|
|
jobject bitmap = NULL;
|
|
CHECK_PTR( ptr, 1, end );
|
|
XP_U8 nCols = *ptr++;
|
|
if ( nCols > 0 ) {
|
|
CHECK_PTR( ptr, 1, end );
|
|
XP_U8 nRows = *ptr++;
|
|
CHECK_PTR( ptr, ((nRows*nCols)+7) / 8, end );
|
|
#ifdef DROP_BITMAPS
|
|
ptr += ((nRows*nCols)+7) / 8;
|
|
#else
|
|
XP_U8 srcByte = 0;
|
|
XP_U8 nBits;
|
|
|
|
jboolean* colors = (jboolean*)XP_CALLOC( ctxt->super.mpool,
|
|
nCols * nRows * sizeof(*colors) );
|
|
jboolean* next = colors;
|
|
|
|
nBits = nRows * nCols;
|
|
for ( int ii = 0; ii < nBits; ++ii ) {
|
|
XP_U8 srcBitIndex = ii % 8;
|
|
XP_U8 srcMask;
|
|
|
|
if ( srcBitIndex == 0 ) {
|
|
srcByte = *ptr++;
|
|
}
|
|
|
|
srcMask = 1 << (7 - srcBitIndex);
|
|
XP_ASSERT( next < (colors + (nRows * nCols)) );
|
|
*next++ = ((srcByte & srcMask) == 0) ? JNI_FALSE : JNI_TRUE;
|
|
}
|
|
|
|
JNIEnv* env = ctxt->env;
|
|
jobject tmp = and_util_makeJBitmap( ctxt->jniutil, nCols, nRows, colors );
|
|
bitmap = (*env)->NewGlobalRef( env, tmp );
|
|
deleteLocalRef( env, tmp );
|
|
XP_FREE( ctxt->super.mpool, colors );
|
|
#endif
|
|
}
|
|
goto done;
|
|
error:
|
|
success = XP_FALSE;
|
|
done:
|
|
*ptrp = ptr;
|
|
*result = bitmap;
|
|
return success;
|
|
} /* andMakeBitmap */
|
|
|
|
static XP_Bool
|
|
andLoadSpecialData( AndDictionaryCtxt* ctxt, XP_U8 const** ptrp,
|
|
const XP_U8* end )
|
|
{
|
|
XP_Bool success = XP_TRUE;
|
|
XP_U16 nSpecials = andCountSpecials( ctxt );
|
|
XP_U8 const* ptr = *ptrp;
|
|
XP_UCHAR** texts;
|
|
XP_UCHAR** textEnds;
|
|
SpecialBitmaps* bitmaps;
|
|
|
|
texts = (XP_UCHAR**)XP_MALLOC( ctxt->super.mpool,
|
|
nSpecials * sizeof(*texts) );
|
|
textEnds = (XP_UCHAR**)XP_MALLOC( ctxt->super.mpool,
|
|
nSpecials * sizeof(*textEnds) );
|
|
|
|
bitmaps = (SpecialBitmaps*)
|
|
XP_CALLOC( ctxt->super.mpool, nSpecials * sizeof(*bitmaps) );
|
|
|
|
for ( Tile ii = 0; ii < ctxt->super.nFaces; ++ii ) {
|
|
|
|
const XP_UCHAR* facep = ctxt->super.facePtrs[(short)ii];
|
|
if ( IS_SPECIAL(*facep) ) {
|
|
/* get the string */
|
|
CHECK_PTR( ptr, 1, end );
|
|
XP_U8 txtlen = *ptr++;
|
|
CHECK_PTR( ptr, txtlen, end );
|
|
XP_UCHAR* text = (XP_UCHAR*)XP_MALLOC(ctxt->super.mpool, txtlen+1);
|
|
texts[(int)*facep] = text;
|
|
textEnds[(int)*facep] = text + txtlen + 1;
|
|
XP_MEMCPY( text, ptr, txtlen );
|
|
ptr += txtlen;
|
|
text[txtlen] = '\0';
|
|
XP_ASSERT( *facep < nSpecials ); /* firing */
|
|
|
|
/* This little hack is safe because all bytes but the first in a
|
|
multi-byte utf-8 char have the high bit set. SYNONYM_DELIM
|
|
does not have its high bit set */
|
|
XP_ASSERT( 0 == (SYNONYM_DELIM & 0x80) );
|
|
for ( ; '\0' != *text; ++text ) {
|
|
if ( *text == SYNONYM_DELIM ) {
|
|
*text = '\0';
|
|
}
|
|
}
|
|
|
|
if ( !andMakeBitmap( ctxt, &ptr, end,
|
|
&bitmaps[(int)*facep].largeBM ) ) {
|
|
goto error;
|
|
}
|
|
if ( !andMakeBitmap( ctxt, &ptr, end,
|
|
&bitmaps[(int)*facep].smallBM ) ) {
|
|
goto error;
|
|
}
|
|
}
|
|
}
|
|
|
|
goto done;
|
|
error:
|
|
success = XP_FALSE;
|
|
done:
|
|
ctxt->super.chars = texts;
|
|
ctxt->super.charEnds = textEnds;
|
|
ctxt->super.bitmaps = bitmaps;
|
|
|
|
*ptrp = ptr;
|
|
return success;
|
|
} /* andLoadSpecialData */
|
|
|
|
/** Android doesn't include iconv for C code to use, so we'll have java do it.
|
|
* Cons up a string with all the tile faces (skipping the specials to make
|
|
* things easier) and have java return an array of strings. Then load one at
|
|
* a time into the expected null-separated format.
|
|
*/
|
|
static void
|
|
splitFaces_via_java( JNIEnv* env, AndDictionaryCtxt* ctxt, const XP_U8* ptr,
|
|
int nFaceBytes, int nFaces, XP_Bool isUTF8 )
|
|
{
|
|
const size_t facesBufSize = nFaces*16;
|
|
XP_UCHAR facesBuf[facesBufSize]; /* seems a reasonable upper bound... */
|
|
int indx = 0;
|
|
int offsets[nFaces];
|
|
int nBytes;
|
|
|
|
jobject jstrarr = and_util_splitFaces( ctxt->jniutil, ptr, nFaceBytes,
|
|
isUTF8 );
|
|
XP_ASSERT( (*env)->GetArrayLength( env, jstrarr ) == nFaces );
|
|
|
|
for ( int ii = 0; ii < nFaces; ++ii ) {
|
|
jobject jstrs = (*env)->GetObjectArrayElement( env, jstrarr, ii );
|
|
offsets[ii] = indx;
|
|
int nAlternates = (*env)->GetArrayLength( env, jstrs );
|
|
for ( int jj = 0; jj < nAlternates; ++jj ) {
|
|
jobject jstr = (*env)->GetObjectArrayElement( env, jstrs, jj );
|
|
nBytes = (*env)->GetStringUTFLength( env, jstr );
|
|
|
|
const char* bytes = (*env)->GetStringUTFChars( env, jstr, NULL );
|
|
char* end;
|
|
long numval = strtol( bytes, &end, 10 );
|
|
if ( end > bytes ) {
|
|
XP_ASSERT( numval < 32 );
|
|
XP_ASSERT( jj == 0 );
|
|
nBytes = 1;
|
|
facesBuf[indx] = (XP_UCHAR)numval;
|
|
} else {
|
|
XP_MEMCPY( &facesBuf[indx], bytes, nBytes );
|
|
}
|
|
(*env)->ReleaseStringUTFChars( env, jstr, bytes );
|
|
deleteLocalRef( env, jstr );
|
|
indx += nBytes;
|
|
facesBuf[indx++] = '\0';
|
|
}
|
|
|
|
deleteLocalRef( env, jstrs );
|
|
XP_ASSERT( indx < facesBufSize );
|
|
}
|
|
deleteLocalRef( env, jstrarr );
|
|
|
|
XP_UCHAR* faces = (XP_UCHAR*)XP_CALLOC( ctxt->super.mpool, indx );
|
|
const XP_UCHAR** ptrs = (const XP_UCHAR**)
|
|
XP_CALLOC( ctxt->super.mpool, nFaces * sizeof(ptrs[0]));
|
|
|
|
XP_MEMCPY( faces, facesBuf, indx );
|
|
for ( int ii = 0; ii < nFaces; ++ii ) {
|
|
ptrs[ii] = &faces[offsets[ii]];
|
|
}
|
|
|
|
XP_ASSERT( !ctxt->super.faces );
|
|
ctxt->super.faces = faces;
|
|
ctxt->super.facesEnd = faces + indx;
|
|
XP_ASSERT( !ctxt->super.facePtrs );
|
|
ctxt->super.facePtrs = ptrs;
|
|
} /* splitFaces_via_java */
|
|
|
|
static XP_UCHAR*
|
|
getNullTermParam( AndDictionaryCtxt* dctx, const XP_U8** ptr,
|
|
XP_U16* headerLen )
|
|
{
|
|
XP_U16 len = 1 + XP_STRLEN( (XP_UCHAR*)*ptr );
|
|
XP_UCHAR* result = XP_MALLOC( dctx->super.mpool, len );
|
|
XP_MEMCPY( result, *ptr, len );
|
|
*ptr += len;
|
|
*headerLen -= len;
|
|
return result;
|
|
}
|
|
|
|
static XP_Bool
|
|
parseDict( AndDictionaryCtxt* ctxt, XP_U8 const* ptr, XP_U32 dictLength,
|
|
XP_U32* numEdges )
|
|
{
|
|
XP_Bool success = XP_TRUE;
|
|
XP_ASSERT( !!ptr );
|
|
const XP_U8* end = ptr + dictLength;
|
|
XP_U32 offset;
|
|
XP_U16 nFaces, numFaceBytes = 0;
|
|
XP_U16 flags;
|
|
void* mappedBase = (void*)ptr;
|
|
XP_U8 nodeSize;
|
|
XP_Bool isUTF8 = XP_FALSE;
|
|
|
|
CHECK_PTR( ptr, sizeof(flags), end );
|
|
flags = n_ptr_tohs( &ptr );
|
|
if ( 0 != (DICT_HEADER_MASK & flags) ) {
|
|
XP_U16 headerLen;
|
|
flags &= ~DICT_HEADER_MASK;
|
|
CHECK_PTR( ptr, sizeof(headerLen), end );
|
|
headerLen = n_ptr_tohs( &ptr );
|
|
if ( 4 <= headerLen ) { /* have word count? */
|
|
CHECK_PTR( ptr, sizeof(ctxt->super.nWords), end );
|
|
ctxt->super.nWords = n_ptr_tohl( &ptr );
|
|
headerLen -= 4; /* don't skip it */
|
|
}
|
|
|
|
if ( 1 <= headerLen ) { /* have description? */
|
|
ctxt->super.desc = getNullTermParam( ctxt, &ptr, &headerLen );
|
|
}
|
|
if ( 1 <= headerLen ) { /* have md5sum? */
|
|
ctxt->super.md5Sum = getNullTermParam( ctxt, &ptr, &headerLen );
|
|
}
|
|
|
|
CHECK_PTR( ptr, headerLen, end );
|
|
ptr += headerLen;
|
|
}
|
|
|
|
flags &= ~DICT_SYNONYMS_MASK;
|
|
if ( flags == 0x0002 ) {
|
|
nodeSize = 3;
|
|
} else if ( flags == 0x0003 ) {
|
|
nodeSize = 4;
|
|
} else if ( flags == 0x0004 ) {
|
|
isUTF8 = XP_TRUE;
|
|
nodeSize = 3;
|
|
} else if ( flags == 0x0005 ) {
|
|
isUTF8 = XP_TRUE;
|
|
nodeSize = 4;
|
|
} else {
|
|
goto error;
|
|
}
|
|
|
|
if ( isUTF8 ) {
|
|
CHECK_PTR( ptr, 1, end );
|
|
numFaceBytes = (XP_U16)(*ptr++);
|
|
}
|
|
CHECK_PTR( ptr, 1, end );
|
|
nFaces = (XP_U16)(*ptr++);
|
|
if ( nFaces > 64 ) {
|
|
goto error;
|
|
}
|
|
|
|
if ( NULL == ctxt->super.md5Sum
|
|
#ifdef DEBUG
|
|
|| XP_TRUE
|
|
#endif
|
|
) {
|
|
JNIEnv* env = ctxt->env;
|
|
jstring jsum = and_util_getMD5SumForDict( ctxt->jniutil,
|
|
ctxt->super.name, NULL, 0 );
|
|
XP_UCHAR* md5Sum = NULL;
|
|
/* If we have a cached sum, check that it's correct. */
|
|
if ( NULL != jsum && NULL != ctxt->super.md5Sum ) {
|
|
md5Sum = getStringCopy( MPPARM(ctxt->super.mpool) env, jsum );
|
|
if ( 0 != XP_STRCMP( ctxt->super.md5Sum, md5Sum ) ) {
|
|
deleteLocalRef( env, jsum );
|
|
jsum = NULL;
|
|
XP_FREE( ctxt->super.mpool, md5Sum );
|
|
md5Sum = NULL;
|
|
}
|
|
}
|
|
|
|
if ( NULL == jsum ) {
|
|
jsum = and_util_getMD5SumForDict( ctxt->jniutil, ctxt->super.name,
|
|
ptr, end - ptr );
|
|
}
|
|
if ( NULL == md5Sum ) {
|
|
md5Sum = getStringCopy( MPPARM(ctxt->super.mpool) env, jsum );
|
|
}
|
|
deleteLocalRef( env, jsum );
|
|
|
|
if ( NULL == ctxt->super.md5Sum ) {
|
|
ctxt->super.md5Sum = md5Sum;
|
|
} else {
|
|
XP_FREE( ctxt->super.mpool, md5Sum );
|
|
}
|
|
}
|
|
|
|
ctxt->super.nodeSize = nodeSize;
|
|
|
|
if ( !isUTF8 ) {
|
|
numFaceBytes = nFaces * 2;
|
|
}
|
|
|
|
ctxt->super.nFaces = (XP_U8)nFaces;
|
|
ctxt->super.isUTF8 = isUTF8;
|
|
|
|
if ( isUTF8 ) {
|
|
CHECK_PTR( ptr, numFaceBytes, end );
|
|
splitFaces_via_java( ctxt->env, ctxt, ptr, numFaceBytes, nFaces,
|
|
XP_TRUE );
|
|
ptr += numFaceBytes;
|
|
} else {
|
|
XP_U8 tmp[nFaces*4]; /* should be enough... */
|
|
XP_U16 nBytes = 0;
|
|
/* Need to translate from iso-8859-n to utf8 */
|
|
CHECK_PTR( ptr, 2 * nFaces, end );
|
|
for ( int ii = 0; ii < nFaces; ++ii ) {
|
|
XP_UCHAR ch = ptr[1];
|
|
|
|
ptr += 2;
|
|
|
|
tmp[nBytes] = ch;
|
|
nBytes += 1;
|
|
}
|
|
XP_ASSERT( nFaces == nBytes );
|
|
splitFaces_via_java( ctxt->env, ctxt, tmp, nBytes, nFaces,
|
|
XP_FALSE );
|
|
}
|
|
|
|
ctxt->super.is_4_byte = (ctxt->super.nodeSize == 4);
|
|
|
|
ctxt->super.countsAndValues =
|
|
(XP_U8*)XP_MALLOC(ctxt->super.mpool, nFaces*2);
|
|
|
|
CHECK_PTR( ptr, 2, end );
|
|
ctxt->super.langCode = ptr[0] & 0x7F;
|
|
ptr += 2; /* skip xloc header */
|
|
CHECK_PTR( ptr, 2 * nFaces, end );
|
|
for ( int ii = 0; ii < nFaces*2; ii += 2 ) {
|
|
ctxt->super.countsAndValues[ii] = *ptr++;
|
|
ctxt->super.countsAndValues[ii+1] = *ptr++;
|
|
}
|
|
|
|
if ( !andLoadSpecialData( ctxt, &ptr, end ) ) {
|
|
goto error;
|
|
}
|
|
|
|
dictLength -= ptr - (XP_U8*)mappedBase;
|
|
if ( dictLength >= sizeof(offset) ) {
|
|
CHECK_PTR( ptr, sizeof(offset), end );
|
|
offset = n_ptr_tohl( &ptr );
|
|
dictLength -= sizeof(offset);
|
|
XP_ASSERT( dictLength % ctxt->super.nodeSize == 0 );
|
|
*numEdges = dictLength / ctxt->super.nodeSize;
|
|
#ifdef DEBUG
|
|
ctxt->super.numEdges = *numEdges;
|
|
#endif
|
|
} else {
|
|
offset = 0;
|
|
}
|
|
|
|
if ( dictLength > 0 ) {
|
|
ctxt->super.base = (array_edge*)ptr;
|
|
ctxt->super.topEdge = ctxt->super.base
|
|
+ (offset * ctxt->super.nodeSize);
|
|
} else {
|
|
ctxt->super.topEdge = (array_edge*)NULL;
|
|
ctxt->super.base = (array_edge*)NULL;
|
|
}
|
|
|
|
setBlankTile( &ctxt->super );
|
|
|
|
goto done;
|
|
error:
|
|
success = XP_FALSE;
|
|
done:
|
|
return success;
|
|
} /* parseDict */
|
|
|
|
static void
|
|
and_dictionary_destroy( DictionaryCtxt* dict )
|
|
{
|
|
AndDictionaryCtxt* ctxt = (AndDictionaryCtxt*)dict;
|
|
XP_LOGF( "%s(dict=%p); code=%x", __func__, ctxt, ctxt->dbgid );
|
|
XP_U16 nSpecials = andCountSpecials( ctxt );
|
|
JNIEnv* env = ctxt->env;
|
|
|
|
if ( !!ctxt->super.chars ) {
|
|
for ( int ii = 0; ii < nSpecials; ++ii ) {
|
|
XP_UCHAR* text = ctxt->super.chars[ii];
|
|
if ( !!text ) {
|
|
XP_FREE( ctxt->super.mpool, text );
|
|
}
|
|
}
|
|
XP_FREE( ctxt->super.mpool, ctxt->super.chars );
|
|
}
|
|
XP_FREEP( ctxt->super.mpool, &ctxt->super.charEnds );
|
|
|
|
if ( !!ctxt->super.bitmaps ) {
|
|
for ( int ii = 0; ii < nSpecials; ++ii ) {
|
|
jobject bitmap = ctxt->super.bitmaps[ii].largeBM;
|
|
if ( !!bitmap ) {
|
|
(*env)->DeleteGlobalRef( env, bitmap );
|
|
}
|
|
}
|
|
XP_FREE( ctxt->super.mpool, ctxt->super.bitmaps );
|
|
}
|
|
|
|
XP_FREEP( ctxt->super.mpool, &ctxt->super.md5Sum );
|
|
XP_FREEP( ctxt->super.mpool, &ctxt->super.desc );
|
|
XP_FREEP( ctxt->super.mpool, &ctxt->super.faces );
|
|
XP_FREEP( ctxt->super.mpool, &ctxt->super.facePtrs );
|
|
XP_FREEP( ctxt->super.mpool, &ctxt->super.countsAndValues );
|
|
XP_FREEP( ctxt->super.mpool, &ctxt->super.name );
|
|
XP_FREEP( ctxt->super.mpool, &ctxt->super.langName );
|
|
|
|
if ( NULL == ctxt->byteArray ) { /* mmap case */
|
|
#ifdef DEBUG
|
|
int err =
|
|
#endif
|
|
munmap( ctxt->bytes, ctxt->bytesSize );
|
|
XP_ASSERT( 0 == err );
|
|
} else {
|
|
(*env)->ReleaseByteArrayElements( env, ctxt->byteArray, ctxt->bytes, 0 );
|
|
(*env)->DeleteGlobalRef( env, ctxt->byteArray );
|
|
}
|
|
XP_FREE( ctxt->super.mpool, ctxt );
|
|
} /* and_dictionary_destroy */
|
|
|
|
jobject
|
|
and_dictionary_getChars( JNIEnv* env, DictionaryCtxt* dict )
|
|
{
|
|
/* XP_ASSERT( env == ((AndDictionaryCtxt*)dict)->env ); */
|
|
/* The above is failing now that dictmgr reuses dicts across threads. I
|
|
think that's ok, that I didn't have a good reason for having this
|
|
assert. But bears watching... */
|
|
|
|
/* This is cheating: specials will be rep'd as 1,2, etc. But as long as
|
|
java code wants to ignore them anyway that's ok. Otherwise need to
|
|
use dict_tilesToString() */
|
|
XP_U16 nFaces = dict_numTileFaces( dict );
|
|
jobject jstrs = makeStringArray( env, nFaces, dict->facePtrs );
|
|
return jstrs;
|
|
}
|
|
|
|
DictionaryCtxt*
|
|
and_dictionary_make_empty( MPFORMAL JNIEnv* env, JNIUtilCtxt* jniutil )
|
|
{
|
|
AndDictionaryCtxt* anddict
|
|
= (AndDictionaryCtxt*)XP_CALLOC( mpool, sizeof( *anddict ) );
|
|
anddict->env = env;
|
|
anddict->jniutil = jniutil;
|
|
#ifdef DEBUG
|
|
anddict->dbgid = rand();
|
|
#endif
|
|
dict_super_init( (DictionaryCtxt*)anddict );
|
|
MPASSIGN( anddict->super.mpool, mpool );
|
|
|
|
LOG_RETURNF( "%p", anddict );
|
|
return (DictionaryCtxt*)anddict;
|
|
}
|
|
|
|
void
|
|
makeDicts( MPFORMAL JNIEnv *env, DictMgrCtxt* dictMgr, JNIUtilCtxt* jniutil,
|
|
DictionaryCtxt** dictp, PlayerDicts* dicts,
|
|
jobjectArray jnames, jobjectArray jdicts, jobjectArray jpaths,
|
|
jstring jlang )
|
|
{
|
|
jsize len = (*env)->GetArrayLength( env, jdicts );
|
|
XP_ASSERT( len == (*env)->GetArrayLength( env, jnames ) );
|
|
|
|
for ( int ii = 0; ii <= MAX_NUM_PLAYERS; ++ii ) {
|
|
DictionaryCtxt* dict = NULL;
|
|
if ( ii < len ) {
|
|
jobject jdict = (*env)->GetObjectArrayElement( env, jdicts, ii );
|
|
jstring jpath = jpaths == NULL ?
|
|
NULL : (*env)->GetObjectArrayElement( env, jpaths, ii );
|
|
if ( NULL != jdict || NULL != jpath ) {
|
|
jstring jname = (*env)->GetObjectArrayElement( env, jnames, ii );
|
|
dict = makeDict( MPPARM(mpool) env, dictMgr, jniutil, jname, jdict,
|
|
jpath, jlang, false );
|
|
XP_ASSERT( !!dict );
|
|
deleteLocalRefs( env, jdict, jname, DELETE_NO_REF );
|
|
}
|
|
deleteLocalRef( env, jpath );
|
|
}
|
|
if ( 0 == ii ) {
|
|
*dictp = dict;
|
|
} else {
|
|
XP_ASSERT( ii-1 < MAX_NUM_PLAYERS );
|
|
dicts->dicts[ii-1] = dict;
|
|
}
|
|
}
|
|
}
|
|
|
|
DictionaryCtxt*
|
|
makeDict( MPFORMAL JNIEnv *env, DictMgrCtxt* dictMgr, JNIUtilCtxt* jniutil, jstring jname,
|
|
jbyteArray jbytes, jstring jpath, jstring jlangname, jboolean check )
|
|
{
|
|
jbyte* bytes = NULL;
|
|
jbyteArray byteArray = NULL;
|
|
off_t bytesSize = 0;
|
|
|
|
const char* name = (*env)->GetStringUTFChars( env, jname, NULL );
|
|
/* remember: dmgr_get calls dict_ref() */
|
|
AndDictionaryCtxt* anddict = (AndDictionaryCtxt*)dmgr_get( dictMgr, name );
|
|
|
|
if ( NULL == anddict ) {
|
|
if ( NULL == jpath ) {
|
|
bytesSize = (*env)->GetArrayLength( env, jbytes );
|
|
byteArray = (*env)->NewGlobalRef( env, jbytes );
|
|
bytes = (*env)->GetByteArrayElements( env, byteArray, NULL );
|
|
} else {
|
|
const char* path = (*env)->GetStringUTFChars( env, jpath, NULL );
|
|
|
|
struct stat statbuf;
|
|
if ( 0 == stat( path, &statbuf ) && 0 < statbuf.st_size ) {
|
|
int fd = open( path, O_RDONLY );
|
|
if ( fd >= 0 ) {
|
|
void* ptr = mmap( NULL, statbuf.st_size, PROT_READ,
|
|
MAP_PRIVATE, fd, 0 );
|
|
close( fd );
|
|
if ( MAP_FAILED != ptr ) {
|
|
bytes = ptr;
|
|
bytesSize = statbuf.st_size;
|
|
}
|
|
}
|
|
}
|
|
(*env)->ReleaseStringUTFChars( env, jpath, path );
|
|
}
|
|
|
|
if ( NULL != bytes ) {
|
|
anddict = (AndDictionaryCtxt*)
|
|
and_dictionary_make_empty( MPPARM(mpool) env, jniutil );
|
|
anddict->bytes = bytes;
|
|
anddict->byteArray = byteArray;
|
|
anddict->bytesSize = bytesSize;
|
|
|
|
anddict->super.destructor = and_dictionary_destroy;
|
|
|
|
/* copy the name */
|
|
anddict->super.name = copyString( mpool, name );
|
|
XP_LOGF( "%s(dict=%p); code=%x; name=%s", __func__, anddict,
|
|
anddict->dbgid, anddict->super.name );
|
|
anddict->super.langName = getStringCopy( MPPARM(mpool)
|
|
env, jlangname );
|
|
|
|
XP_U32 numEdges = 0;
|
|
XP_Bool parses = parseDict( anddict, (XP_U8*)anddict->bytes,
|
|
bytesSize, &numEdges );
|
|
if ( !parses || (check && !checkSanity( &anddict->super,
|
|
numEdges ) ) ) {
|
|
and_dictionary_destroy( (DictionaryCtxt*)anddict );
|
|
anddict = NULL;
|
|
}
|
|
}
|
|
dmgr_put( dictMgr, name, &anddict->super );
|
|
dict_ref( &anddict->super );
|
|
}
|
|
|
|
(*env)->ReleaseStringUTFChars( env, jname, name );
|
|
|
|
return &anddict->super;
|
|
} /* makeDict */
|
|
|
|
#ifdef DEBUG
|
|
uint32_t
|
|
andDictID( const DictionaryCtxt* dict )
|
|
{
|
|
const AndDictionaryCtxt* ctxt = (AndDictionaryCtxt*)dict;
|
|
return ctxt->dbgid;
|
|
}
|
|
#endif
|