/* -*- 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 #include #include #include #include #include #include #include #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; 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, XWEnv xwe, const XP_U8* bytes, XP_U16 nBytes, XP_U16 nFaces ) { AndDictionaryCtxt* ctxt = (AndDictionaryCtxt*)dict; splitFaces_via_java( xwe, 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 ) { XP_UCHAR facesBuf[nFaces*16]; /* 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 < VSIZE(facesBuf) ); } 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, XWEnv xwe, 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; } JNIEnv* env = xwe; if ( NULL == ctxt->super.md5Sum #ifdef DEBUG || XP_TRUE #endif ) { 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( 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( 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, XWEnv xwe ) { AndDictionaryCtxt* ctxt = (AndDictionaryCtxt*)dict; XP_LOGF( "%s(dict=%p); code=%x", __func__, ctxt, ctxt->dbgid ); XP_U16 nSpecials = andCountSpecials( ctxt ); JNIEnv* env = xwe; 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 JNIUtilCtxt* jniutil ) { AndDictionaryCtxt* anddict = (AndDictionaryCtxt*)XP_CALLOC( mpool, sizeof( *anddict ) ); 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 <= VSIZE(dicts->dicts); ++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 < VSIZE( dicts->dicts ) ); 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, env, 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) 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, env, (XP_U8*)anddict->bytes, bytesSize, &numEdges ); if ( !parses || (check && !checkSanity( &anddict->super, numEdges ) ) ) { and_dictionary_destroy( (DictionaryCtxt*)anddict, env ); anddict = NULL; } } dmgr_put( dictMgr, env, name, &anddict->super ); dict_ref( &anddict->super, env ); } (*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