mirror of
git://xwords.git.sourceforge.net/gitroot/xwords/xwords
synced 2025-01-22 07:28:16 +01:00
600 lines
18 KiB
C
600 lines
18 KiB
C
/* -*-mode: C; fill-column: 78; c-basic-offset: 4; -*- */
|
|
/****************************************************************************
|
|
* *
|
|
* Copyright 1999 - 2003 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 <PalmTypes.h>
|
|
#include <Form.h>
|
|
#include <VFSMgr.h>
|
|
#include <FeatureMgr.h>
|
|
|
|
#include "callback.h"
|
|
#include "dictlist.h"
|
|
#include "palmmain.h"
|
|
#include "palmutil.h"
|
|
#include "palmdict.h"
|
|
#include "strutils.h"
|
|
#include "xwords4defines.h"
|
|
#include "LocalizedStrIncludes.h"
|
|
|
|
#define TYPE_DAWG 'DAWG'
|
|
#ifdef NODE_CAN_4
|
|
# define TYPE_XWRDICT 'XwrD'
|
|
#else
|
|
# define TYPE_XWRDICT 'Xwr3'
|
|
#endif
|
|
|
|
//////////////////////////////////////////////////////////////////////////////
|
|
// typedef and #defines
|
|
//////////////////////////////////////////////////////////////////////////////
|
|
|
|
struct PalmDictList {
|
|
XP_U16 nDicts;
|
|
DictListEntry dictArray[1];
|
|
};
|
|
|
|
//////////////////////////////////////////////////////////////////////////////
|
|
// Prototypes
|
|
//////////////////////////////////////////////////////////////////////////////
|
|
static PalmDictList* dictListMakePriv( MPFORMAL XP_U32 creatorSought,
|
|
XP_U16 versSought );
|
|
|
|
|
|
XP_Bool
|
|
getNthDict( const PalmDictList* dl, short n, DictListEntry** dle )
|
|
{
|
|
XP_Bool exists = !!dl && (dl->nDicts > n);
|
|
if ( exists ) {
|
|
*dle = (DictListEntry*)&dl->dictArray[n];
|
|
}
|
|
return exists;
|
|
} /* getNthDict */
|
|
|
|
XP_Bool
|
|
getDictWithName( const PalmDictList* dl, const XP_UCHAR* name,
|
|
DictListEntry** dlep )
|
|
{
|
|
XP_Bool result = XP_FALSE;
|
|
|
|
if ( !!dl ) {
|
|
XP_U16 i;
|
|
XP_UCHAR* extName;
|
|
XP_UCHAR oldChName = '\0'; /* shut compiler up */
|
|
DictListEntry* dle = (DictListEntry*)dl->dictArray;
|
|
|
|
extName = (XP_UCHAR*)StrStr((const char*)name,
|
|
(const char*)".pdb" );
|
|
|
|
if ( !!extName ) {
|
|
oldChName = *extName;
|
|
*extName = '\0';
|
|
}
|
|
|
|
for ( i = 0; !result && i < dl->nDicts; ++i ) {
|
|
XP_UCHAR* extCand;
|
|
XP_UCHAR oldChCand = '\0';
|
|
|
|
extCand = (XP_UCHAR*)StrStr((const char*)dle->baseName, ".pdb" );
|
|
if ( !!extCand ) {
|
|
oldChCand = *extCand;
|
|
*extCand = '\0';
|
|
}
|
|
|
|
if ( 0 == XP_STRCMP( (const char*)name,
|
|
(const char*)dle->baseName ) ) {
|
|
|
|
*dlep = dle;
|
|
result = XP_TRUE;
|
|
}
|
|
|
|
if ( !!extCand ) {
|
|
*extCand = oldChCand;
|
|
}
|
|
|
|
++dle;
|
|
}
|
|
|
|
if ( !!extName ) {
|
|
*extName = oldChName;
|
|
}
|
|
|
|
}
|
|
return result;
|
|
} /* getDictWithName */
|
|
|
|
void
|
|
cacheDictForName( PalmDictList* dl, const XP_UCHAR* dictName,
|
|
DictionaryCtxt* dict )
|
|
{
|
|
DictListEntry* dle;
|
|
(void)getDictWithName(dl, dictName, &dle );
|
|
XP_ASSERT( getDictWithName(dl, dictName, &dle ) );
|
|
XP_ASSERT( !dle->dict );
|
|
|
|
dle->dict = dict;
|
|
} /* cacheDictForName */
|
|
|
|
void
|
|
removeFromDictCache( PalmDictList* dl, XP_UCHAR* dictName )
|
|
{
|
|
DictListEntry* dle;
|
|
(void)getDictWithName( dl, dictName, &dle );
|
|
XP_ASSERT( getDictWithName( dl, dictName, &dle ) );
|
|
XP_ASSERT( !!dle->dict );
|
|
|
|
dle->dict = NULL;
|
|
} /* removeFromDictCache */
|
|
|
|
static XP_Bool
|
|
addEntry( MPFORMAL PalmDictList** dlp, DictListEntry* dle )
|
|
{
|
|
XP_Bool isNew;
|
|
PalmDictList* dl = *dlp;
|
|
DictListEntry* ignore;
|
|
|
|
if ( !dl ) {
|
|
dl = (PalmDictList*)XP_MALLOC( mpool, sizeof(*dl) );
|
|
XP_MEMSET( dl, 0, sizeof(*dl) );
|
|
}
|
|
|
|
isNew = !getDictWithName( dl, dle->baseName, &ignore );
|
|
if ( isNew ) {
|
|
XP_U16 size = sizeof(*dl);
|
|
size += dl->nDicts * sizeof( dl->dictArray[0] );
|
|
|
|
dl = (PalmDictList*)XP_REALLOC( mpool, (XP_U8*)dl, size );
|
|
|
|
dle->dict = NULL;
|
|
XP_MEMCPY( &dl->dictArray[dl->nDicts++], dle,
|
|
sizeof( dl->dictArray[0] ) );
|
|
}
|
|
|
|
*dlp = dl;
|
|
return isNew;
|
|
} /* addEntry */
|
|
|
|
static void
|
|
searchDir( MPFORMAL PalmDictList** dlp, UInt16 volNum,
|
|
unsigned char XP_UNUSED(separator),
|
|
unsigned char* path, XP_U16 pathSize, XP_U32 creatorSought,
|
|
XP_U16 versSought )
|
|
{
|
|
Err err;
|
|
FileRef dirRef;
|
|
XP_U16 pathLen = XP_STRLEN( (const char*)path );
|
|
|
|
err = VFSFileOpen( volNum, (const char*)path, vfsModeRead, &dirRef );
|
|
if ( err == errNone ) {
|
|
UInt32 dEnum = vfsIteratorStart;
|
|
FileInfoType fit;
|
|
|
|
fit.nameP = (char*)path + pathLen;
|
|
|
|
while ( dEnum != vfsIteratorStop ) {
|
|
XP_UCHAR* ext;
|
|
fit.nameBufLen = pathSize - pathLen;
|
|
err = VFSDirEntryEnumerate( dirRef, &dEnum, &fit );
|
|
|
|
if ( err != errNone ) {
|
|
break;
|
|
}
|
|
|
|
if ( (fit.attributes & vfsFileAttrDirectory) != 0 ) {
|
|
#ifdef RECURSIVE_VFS_SEARCH
|
|
XP_U16 len = XP_STRLEN((const char*)path);
|
|
path[len] = separator;
|
|
path[len+1] = '\0';
|
|
searchDir( MPPARM(mpool) dlp, volNum, separator,
|
|
path, pathSize, creatorSought, versSought );
|
|
#endif
|
|
} else if ( (ext = (XP_UCHAR*)StrStr( (const char*)path, ".pdb" ))
|
|
!= NULL ) {
|
|
|
|
/* find out if it's a crosswords dict. */
|
|
FileRef fileRef;
|
|
UInt32 type, creator;
|
|
|
|
err = VFSFileOpen( volNum, (const char*)path, vfsModeRead,
|
|
&fileRef );
|
|
if ( err == errNone ) {
|
|
XP_U16 vers;
|
|
err = VFSFileDBInfo( fileRef, NULL, /* name */
|
|
NULL, /* attributes */
|
|
&vers, /* versionP */
|
|
NULL, /* crDateP */
|
|
NULL, NULL, /*UInt32 *modDateP, UInt32 *bckUpDateP,*/
|
|
NULL, NULL, /*UInt32 *modNumP, MemHandle *appInfoHP,*/
|
|
NULL, /*MemHandle *sortInfoHP, */
|
|
&type, &creator,
|
|
NULL ); /* nRecords */
|
|
VFSFileClose( fileRef );
|
|
|
|
if ( (err == errNone) && (type == TYPE_DAWG) &&
|
|
(creator == creatorSought) && (vers == versSought) ) {
|
|
DictListEntry dl;
|
|
|
|
dl.path = copyString( mpool, path );
|
|
dl.location = DL_VFS;
|
|
dl.u.vfsData.volNum = volNum;
|
|
dl.baseName = dl.path + pathLen;
|
|
|
|
if ( !addEntry( MPPARM(mpool) dlp, &dl ) ) {
|
|
XP_FREE( mpool, dl.path );
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
path[pathLen] = '\0';
|
|
VFSFileClose( dirRef );
|
|
}
|
|
|
|
} /* searchDir */
|
|
|
|
static void
|
|
tryVFSSearch( MPFORMAL PalmDictList** dlp, XP_U32 creatorSought,
|
|
XP_U16 versSought )
|
|
{
|
|
Err err;
|
|
UInt16 volNum;
|
|
UInt32 vEnum;
|
|
|
|
vEnum = vfsIteratorStart;
|
|
while ( vEnum != vfsIteratorStop ) {
|
|
unsigned char pathStr[265];
|
|
MemHandle h;
|
|
UInt16 bufLen;
|
|
|
|
err = VFSVolumeEnumerate( &volNum, &vEnum );
|
|
if ( err != errNone ) {
|
|
break;
|
|
}
|
|
|
|
/* Search from the default (/palm/Launcher, normally) */
|
|
bufLen = sizeof(pathStr);
|
|
err = VFSGetDefaultDirectory( volNum, ".pdb", (char*)pathStr,
|
|
&bufLen );
|
|
if ( err == errNone ) {
|
|
searchDir( MPPARM(mpool) dlp, volNum, pathStr[0],
|
|
pathStr, sizeof(pathStr), creatorSought, versSought );
|
|
}
|
|
|
|
h = DmGetResource( 'tSTR', 1000 );
|
|
if ( !!h ) {
|
|
pathStr[0] = '\0';
|
|
XP_STRCAT( pathStr, MemHandleLock(h) );
|
|
MemHandleUnlock( h );
|
|
DmReleaseResource( h );
|
|
searchDir( MPPARM(mpool) dlp, volNum, pathStr[0],
|
|
pathStr, sizeof(pathStr), creatorSought, versSought );
|
|
}
|
|
}
|
|
|
|
} /* tryVFSSearch */
|
|
|
|
/* if we've allocated extra space in the array as an optimization now's when
|
|
* we pull back */
|
|
static void
|
|
cleanList( PalmDictList** XP_UNUSED(dl) )
|
|
{
|
|
} /* cleanList */
|
|
|
|
PalmDictList*
|
|
DictListMake( MPFORMAL_NOCOMMA )
|
|
{
|
|
return dictListMakePriv( MPPARM(mpool) TYPE_XWRDICT, 1 );
|
|
}
|
|
|
|
static PalmDictList*
|
|
dictListMakePriv( MPFORMAL XP_U32 creatorSought, XP_U16 versSought )
|
|
{
|
|
Err err;
|
|
DmSearchStateType stateType;
|
|
UInt32 vers;
|
|
PalmDictList* dl = NULL;
|
|
UInt16 cardNo;
|
|
LocalID dbID;
|
|
Boolean newSearch = true;
|
|
XP_Bool found = false;
|
|
|
|
/* first the DM case */
|
|
while ( !found ) {
|
|
err = DmGetNextDatabaseByTypeCreator( newSearch, &stateType, TYPE_DAWG,
|
|
creatorSought,
|
|
false,// onlyLatestVers,
|
|
&cardNo, &dbID );
|
|
if ( err != errNone ) {
|
|
break;
|
|
} else {
|
|
XP_UCHAR nameBuf[33];
|
|
XP_U16 nameLen;
|
|
DictListEntry dle;
|
|
XP_U16 vers;
|
|
|
|
err = DmDatabaseInfo( cardNo, dbID, (char*)nameBuf, NULL, &vers,
|
|
NULL, NULL, NULL, NULL, NULL, NULL, NULL,
|
|
NULL );
|
|
if ( (err == errNone) && (vers == versSought) ) {
|
|
nameLen = XP_STRLEN( (const char*)nameBuf ) + 1;
|
|
|
|
dle.location = DL_STORAGE;
|
|
dle.u.dmData.cardNo = cardNo;
|
|
dle.u.dmData.dbID = dbID;
|
|
dle.path = dle.baseName = XP_MALLOC( mpool, nameLen );
|
|
XP_MEMCPY( dle.path, nameBuf, nameLen );
|
|
|
|
addEntry( MPPARM(mpool) &dl, &dle );
|
|
}
|
|
}
|
|
|
|
newSearch = false;
|
|
}
|
|
|
|
/* then the VFS case */
|
|
err = FtrGet( sysFileCVFSMgr, vfsFtrIDVersion, &vers );
|
|
if ( err == errNone ) {
|
|
tryVFSSearch( MPPARM(mpool) &dl, creatorSought, versSought );
|
|
}
|
|
|
|
cleanList( &dl );
|
|
|
|
return dl;
|
|
} /* dictListMakePriv */
|
|
|
|
void
|
|
DictListFree( MPFORMAL PalmDictList* dl )
|
|
{
|
|
if ( !!dl ) {
|
|
DictListEntry* dle = dl->dictArray;
|
|
XP_U16 i;
|
|
|
|
for ( i = 0; i < dl->nDicts; ++i, ++dle ) {
|
|
XP_FREE( mpool, dle->path );
|
|
}
|
|
|
|
XP_FREE( mpool, dl );
|
|
}
|
|
} /* dictListFree */
|
|
|
|
XP_U16
|
|
DictListCount( PalmDictList* dl )
|
|
{
|
|
XP_U16 result;
|
|
if ( !dl ) {
|
|
result = 0;
|
|
} else {
|
|
result = dl->nDicts;
|
|
}
|
|
return result;
|
|
} /* dictListCount */
|
|
|
|
#ifdef NODE_CAN_4
|
|
|
|
/*****************************************************************************
|
|
* Conversion from old Crosswords DAWG format to new. It should be possible
|
|
* for this to go away once the new format's been out for a while.
|
|
*****************************************************************************/
|
|
|
|
static XP_Bool
|
|
convertOneRecord( DmOpenRef ref, XP_U16 index )
|
|
{
|
|
XP_Bool success = XP_FALSE;
|
|
MemHandle h = DmGetRecord( ref, index );
|
|
XP_U16 siz = MemHandleSize( h );
|
|
XP_U8* recPtr;
|
|
XP_U16 i;
|
|
Err err;
|
|
|
|
XP_U16 nRecs = siz / 3;
|
|
XP_U8* tmp = MemPtrNew( siz );
|
|
|
|
XP_ASSERT( !!tmp );
|
|
XP_ASSERT( (siz % 3) == 0 );
|
|
|
|
recPtr = MemHandleLock(h);
|
|
XP_MEMCPY( tmp, recPtr, siz );
|
|
|
|
for ( i = 0; i < nRecs; ++i ) {
|
|
array_edge_old* edge = (array_edge_old*)&tmp[i * 3];
|
|
XP_U8 oldBits = edge->bits;
|
|
XP_U8 newBits = 0;
|
|
|
|
XP_ASSERT( LETTERMASK_OLD == LETTERMASK_NEW_3 );
|
|
XP_ASSERT( LASTEDGEMASK_OLD == LASTEDGEMASK_NEW );
|
|
newBits |= (oldBits & (LETTERMASK_OLD | LASTEDGEMASK_OLD) );
|
|
|
|
if ( (oldBits & ACCEPTINGMASK_OLD) != 0 ) {
|
|
newBits |= ACCEPTINGMASK_NEW;
|
|
}
|
|
|
|
if ( (oldBits & EXTRABITMASK_OLD) != 0 ) {
|
|
newBits |= EXTRABITMASK_NEW;
|
|
}
|
|
|
|
edge->bits = newBits;
|
|
}
|
|
|
|
err = DmWrite( recPtr, 0, tmp, siz );
|
|
XP_ASSERT( err == errNone );
|
|
success = err == errNone;
|
|
|
|
MemPtrFree( tmp );
|
|
MemHandleUnlock( h );
|
|
DmReleaseRecord( ref, index, true );
|
|
return success;
|
|
} /* convertOneRecord */
|
|
|
|
static XP_Bool
|
|
convertOneDict( UInt16 cardNo, LocalID dbID )
|
|
{
|
|
Err err;
|
|
UInt32 creator;
|
|
DmOpenRef ref;
|
|
MemHandle h;
|
|
dawg_header* header;
|
|
dawg_header tmp;
|
|
XP_U16 siz;
|
|
unsigned char charTableRecNum, firstEdgeRecNum;
|
|
XP_U16 nChars;
|
|
|
|
/* now modify the flags */
|
|
ref = DmOpenDatabase( cardNo, dbID, dmModeReadWrite );
|
|
XP_ASSERT( ref != 0 );
|
|
h = DmGetRecord( ref, 0 );
|
|
siz = MemHandleSize( h );
|
|
if ( siz < sizeof(*header) ) {
|
|
MemHandleResize( h, sizeof(*header) );
|
|
}
|
|
|
|
tmp.flags = XP_HTONS(0x0002);
|
|
XP_ASSERT( sizeof(tmp.flags) == 2 );
|
|
header = (dawg_header*)MemHandleLock(h);
|
|
charTableRecNum = header->charTableRecNum;
|
|
firstEdgeRecNum = header->firstEdgeRecNum;
|
|
DmWrite( header, OFFSET_OF(dawg_header,flags), &tmp.flags,
|
|
sizeof(tmp.flags) );
|
|
MemHandleUnlock(h);
|
|
DmReleaseRecord( ref, 0, true );
|
|
|
|
/* Now convert to 16-bit psuedo-unicode */
|
|
h = DmGetRecord( ref, charTableRecNum );
|
|
XP_ASSERT( !!h );
|
|
nChars = MemHandleSize( h );
|
|
err = MemHandleResize( h, nChars * 2 );
|
|
XP_ASSERT( err == errNone );
|
|
|
|
if ( err == errNone ) {
|
|
XP_U8 buf[(MAX_UNIQUE_TILES+1)*2];
|
|
XP_S16 i;
|
|
XP_U8* ptr = (XP_U8*)MemHandleLock( h );
|
|
|
|
XP_MEMSET( buf, 0, sizeof(buf) );
|
|
|
|
for ( i = 0; i < nChars; ++i ) {
|
|
buf[(i*2)+1] = ptr[i];
|
|
}
|
|
DmWrite( ptr, 0, buf, nChars * 2 );
|
|
MemHandleUnlock(h);
|
|
}
|
|
err = DmReleaseRecord( ref, charTableRecNum, true );
|
|
XP_ASSERT( err == errNone );
|
|
|
|
/* Now transpose the accepting and extra bits for every node. */
|
|
if ( err == errNone ) {
|
|
XP_U32 nRecords = DmNumRecords(ref);
|
|
XP_U16 i;
|
|
|
|
for ( i = firstEdgeRecNum; i < nRecords; ++i ) {
|
|
convertOneRecord( ref, i );
|
|
}
|
|
}
|
|
|
|
err = DmCloseDatabase( ref );
|
|
XP_ASSERT( err == errNone );
|
|
|
|
if ( err == errNone ) {
|
|
XP_U16 newVers = 1;
|
|
creator = TYPE_XWRDICT;
|
|
err = DmSetDatabaseInfo( cardNo, dbID, NULL,
|
|
NULL, &newVers, NULL,
|
|
NULL, NULL,
|
|
NULL, NULL,
|
|
NULL, NULL,
|
|
&creator );
|
|
XP_ASSERT( err == errNone );
|
|
}
|
|
return err == errNone;
|
|
} /* convertOneDict */
|
|
|
|
static XP_Bool
|
|
confirmDictConvert( PalmAppGlobals* globals, const XP_UCHAR* name )
|
|
{
|
|
XP_UCHAR buf[128];
|
|
const XP_UCHAR *fmt = getResString( globals, STRS_CONFIRM_ONEDICT );
|
|
XP_ASSERT( !!fmt );
|
|
XP_SNPRINTF( buf, sizeof(buf), fmt, name );
|
|
return palmask( globals, buf, NULL, -1 );
|
|
} /* confirmDictConvert */
|
|
|
|
void
|
|
offerConvertOldDicts( PalmAppGlobals* globals )
|
|
{
|
|
PalmDictList* dl = dictListMakePriv( MPPARM(globals->mpool) 'Xwr3', 0 );
|
|
XP_U16 count = DictListCount(dl);
|
|
Err err;
|
|
|
|
if ( count > 0 && palmaskFromStrId( globals, STR_CONFIRM_CONVERTDICT,
|
|
-1 ) ) {
|
|
|
|
XP_U16 i;
|
|
for ( i = 0; i < count; ++i ) {
|
|
DictListEntry* dle;
|
|
if ( getNthDict( dl, i, &dle ) ) {
|
|
|
|
if ( dle->location == DL_STORAGE ) {
|
|
|
|
if ( confirmDictConvert( globals, dle->baseName ) ) {
|
|
convertOneDict( dle->u.dmData.cardNo,
|
|
dle->u.dmData.dbID );
|
|
}
|
|
|
|
} else {
|
|
|
|
if ( confirmDictConvert( globals, dle->baseName ) ) {
|
|
|
|
UInt16 cardNo;
|
|
LocalID dbID;
|
|
UInt16 volRefNum = dle->u.vfsData.volNum;
|
|
|
|
XP_ASSERT( dle->location == DL_VFS );
|
|
|
|
XP_LOGF( "trying %s", dle->path );
|
|
|
|
/* copy from SD card to storage, convert, copy back */
|
|
err = VFSImportDatabaseFromFile( volRefNum,
|
|
(const char*)dle->path,
|
|
&cardNo, &dbID );
|
|
XP_LOGF( "VFSImportDatabaseFromFile => %d", err );
|
|
if ( err == errNone && convertOneDict( cardNo, dbID ) ) {
|
|
|
|
err = VFSFileDelete( volRefNum, dle->path );
|
|
XP_LOGF( "VFSFileDelete=>%d", err );
|
|
if ( err == errNone ) {
|
|
|
|
err = VFSExportDatabaseToFile( volRefNum,
|
|
(const char*)dle->path,
|
|
cardNo, dbID );
|
|
|
|
XP_LOGF( "VFSExportDatabaseToFile => %d", err );
|
|
|
|
XP_ASSERT( err == errNone );
|
|
err = DmDeleteDatabase( cardNo, dbID );
|
|
XP_ASSERT( err == errNone );
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
DictListFree( MPPARM(globals->mpool) dl );
|
|
} /* offerConvertOldDicts */
|
|
#endif
|