517 lines
16 KiB
C
517 lines
16 KiB
C
/*++
|
|
|
|
Copyright (c) 1990-1999 Microsoft Corporation, All Rights Reserved
|
|
|
|
Module Name:
|
|
|
|
DIC.c
|
|
|
|
++*/
|
|
#include <windows.h>
|
|
#include <winerror.h>
|
|
#include <immdev.h>
|
|
#include "imeattr.h"
|
|
#include "imedefs.h"
|
|
#include "imerc.h"
|
|
|
|
#if !defined(ROMANIME)
|
|
#if !defined(WINIME) && !defined(UNICDIME)
|
|
/**********************************************************************/
|
|
/* MemoryLess() */
|
|
/**********************************************************************/
|
|
void PASCAL MemoryLess(
|
|
#if defined(UNIIME)
|
|
LPINSTDATAL lpInstL,
|
|
LPIMEL lpImeL,
|
|
#endif
|
|
DWORD fdwErrMsg)
|
|
{
|
|
TCHAR szErrMsg[64];
|
|
|
|
if (lpImeL->fdwErrMsg & fdwErrMsg) {
|
|
// message already prompted
|
|
return;
|
|
}
|
|
|
|
LoadString(hInst, IDS_MEM_LESS_ERR, szErrMsg, sizeof(szErrMsg)/sizeof(TCHAR));
|
|
|
|
lpImeL->fdwErrMsg |= fdwErrMsg;
|
|
MessageBeep((UINT)-1);
|
|
MessageBox((HWND)NULL, szErrMsg, lpImeL->szIMEName,
|
|
MB_OK|MB_ICONHAND|MB_TASKMODAL|MB_TOPMOST);
|
|
|
|
return;
|
|
}
|
|
|
|
/**********************************************************************/
|
|
/* ReadUsrDicToMem() */
|
|
/* Return Value: */
|
|
/* TRUE - successful, FALSE - failure */
|
|
/**********************************************************************/
|
|
BOOL PASCAL ReadUsrDicToMem(
|
|
#if defined(UNIIME)
|
|
LPINSTDATAL lpInstL,
|
|
LPIMEL lpImeL,
|
|
#endif
|
|
HANDLE hUsrDicFile,
|
|
DWORD dwUsrDicSize,
|
|
UINT uUsrDicSize,
|
|
UINT uRecLen,
|
|
UINT uReadLen,
|
|
UINT uWriteLen)
|
|
{
|
|
LPBYTE lpUsrDicMem, lpMem, lpMemLimit;
|
|
DWORD dwPos, dwReadByte;
|
|
|
|
if (dwUsrDicSize < 258) { // no char in this dictionary
|
|
return (TRUE);
|
|
}
|
|
|
|
lpUsrDicMem = MapViewOfFile(lpInstL->hUsrDicMem, FILE_MAP_WRITE, 0, 0,
|
|
uUsrDicSize + 20);
|
|
|
|
if (!lpUsrDicMem) {
|
|
CloseHandle(lpInstL->hUsrDicMem);
|
|
MemoryLess(
|
|
#if defined(UNIIME)
|
|
lpInstL, lpImeL,
|
|
#endif
|
|
ERRMSG_MEM_USRDIC);
|
|
lpInstL->hUsrDicMem = NULL;
|
|
return (FALSE);
|
|
}
|
|
|
|
lpMemLimit = lpUsrDicMem + uUsrDicSize;
|
|
|
|
// read in data, skip header - two headers are similiar
|
|
dwPos = SetFilePointer(hUsrDicFile, 258, (LPLONG)NULL, FILE_BEGIN);
|
|
|
|
for (lpMem = lpUsrDicMem; dwPos < dwUsrDicSize; lpMem += uWriteLen) {
|
|
short i;
|
|
DWORD dwPattern;
|
|
BOOL retVal;
|
|
|
|
if (lpMem >= lpMemLimit) {
|
|
break;
|
|
}
|
|
|
|
retVal = ReadFile(hUsrDicFile, lpMem, uReadLen, &dwReadByte,
|
|
(LPOVERLAPPED)NULL);
|
|
|
|
if ( retVal == FALSE )
|
|
{
|
|
UnmapViewOfFile(lpUsrDicMem);
|
|
CloseHandle(lpInstL->hUsrDicMem);
|
|
MemoryLess(
|
|
#if defined(UNIIME)
|
|
lpInstL, lpImeL,
|
|
#endif
|
|
ERRMSG_MEM_USRDIC);
|
|
lpInstL->hUsrDicMem = NULL;
|
|
return (FALSE);
|
|
}
|
|
|
|
// Compress the sequence code and put the first char most significant.
|
|
// Limitation - 32 bits only
|
|
|
|
dwPattern = 0;
|
|
|
|
for (i = 0; i < lpImeL->nMaxKey; i++) {
|
|
dwPattern <<= lpImeL->nSeqBits;
|
|
dwPattern |= *(lpMem + 2 + i);
|
|
}
|
|
|
|
*(LPUNADWORD)(lpMem + 2) = dwPattern;
|
|
|
|
// go to next record
|
|
dwPos = SetFilePointer(hUsrDicFile, dwPos + uRecLen, (LPLONG)NULL,
|
|
FILE_BEGIN);
|
|
}
|
|
|
|
UnmapViewOfFile(lpUsrDicMem);
|
|
|
|
return (TRUE);
|
|
}
|
|
|
|
/**********************************************************************/
|
|
/* LoadUsrDicFile() */
|
|
/* Description: */
|
|
/* try to convert to sequence code format, compression and */
|
|
/* don't use two way to search */
|
|
/**********************************************************************/
|
|
void PASCAL LoadUsrDicFile( // load user dic file into memory
|
|
LPINSTDATAL lpInstL,
|
|
LPIMEL lpImeL)
|
|
{
|
|
HANDLE hReadUsrDicMem;
|
|
HANDLE hUsrDicFile;
|
|
DWORD dwUsrDicFileSize;
|
|
UINT uRecLen, uReadLen, uWriteLen;
|
|
UINT uUsrDicSize;
|
|
BOOL fRet;
|
|
|
|
// no user dictionary
|
|
if (!lpImeL->szUsrDicMap[0]) {
|
|
lpImeL->uUsrDicSize = 0;
|
|
CloseHandle(lpInstL->hUsrDicMem);
|
|
lpInstL->hUsrDicMem = NULL;
|
|
lpImeL->fdwErrMsg &= ~(ERRMSG_LOAD_USRDIC | ERRMSG_MEM_USRDIC);
|
|
return;
|
|
}
|
|
|
|
if (lpInstL->hUsrDicMem) {
|
|
// the memory is already here
|
|
goto LoadUsrDicErrMsg;
|
|
}
|
|
|
|
hReadUsrDicMem = OpenFileMapping(FILE_MAP_READ, FALSE,
|
|
lpImeL->szUsrDicMap);
|
|
|
|
if (hReadUsrDicMem) {
|
|
// another process already create a mapping file, we will use it
|
|
goto LoadUsrDicMem;
|
|
}
|
|
|
|
// read the user dic file into memory
|
|
hUsrDicFile = CreateFile(lpImeL->szUsrDic, GENERIC_READ,
|
|
FILE_SHARE_READ|FILE_SHARE_WRITE,
|
|
NULL, OPEN_EXISTING,
|
|
FILE_ATTRIBUTE_NORMAL, (HANDLE)NULL);
|
|
|
|
if (hUsrDicFile != INVALID_HANDLE_VALUE) { // OK
|
|
goto OpenUsrDicFile;
|
|
}
|
|
|
|
// if the work station version, SHARE_WRITE may fail
|
|
hUsrDicFile = CreateFile(lpImeL->szUsrDic, GENERIC_READ,
|
|
FILE_SHARE_READ,
|
|
NULL, OPEN_EXISTING,
|
|
FILE_ATTRIBUTE_NORMAL, (HANDLE)NULL);
|
|
|
|
OpenUsrDicFile:
|
|
if (hUsrDicFile != INVALID_HANDLE_VALUE) { // OK
|
|
lpImeL->fdwErrMsg &= ~(ERRMSG_LOAD_USRDIC);
|
|
} else if (lpImeL->fdwErrMsg & ERRMSG_LOAD_USRDIC) {
|
|
// already prompt error message before, no more
|
|
return;
|
|
} else {
|
|
TCHAR szFmtStr[64];
|
|
TCHAR szErrMsg[2 * MAX_PATH];
|
|
HRESULT hr;
|
|
|
|
// temp use szIMEName as format string buffer of error message
|
|
LoadString(hInst, IDS_FILE_OPEN_ERR, szFmtStr, sizeof(szFmtStr)/sizeof(TCHAR));
|
|
hr = StringCchPrintf(szErrMsg, ARRAYSIZE(szErrMsg), szFmtStr, lpImeL->szUsrDic);
|
|
if (FAILED(hr))
|
|
return;
|
|
|
|
lpImeL->fdwErrMsg |= ERRMSG_LOAD_USRDIC;
|
|
MessageBeep((UINT)-1);
|
|
MessageBox((HWND)NULL, szErrMsg, lpImeL->szIMEName,
|
|
MB_OK|MB_ICONHAND|MB_TASKMODAL|MB_TOPMOST);
|
|
return;
|
|
}
|
|
|
|
// one record length - only sequence code, now
|
|
uRecLen = lpImeL->nMaxKey + 4;
|
|
// read sequence code and internal code
|
|
uReadLen = lpImeL->nMaxKey + 2;
|
|
// length write into memory handle
|
|
uWriteLen = lpImeL->nSeqBytes + 2;
|
|
|
|
// get the length of the file
|
|
dwUsrDicFileSize = GetFileSize(hUsrDicFile, (LPDWORD)NULL);
|
|
uUsrDicSize = (UINT)(dwUsrDicFileSize - 256) / uRecLen * uWriteLen;
|
|
|
|
// max EUDC chars
|
|
lpInstL->hUsrDicMem = CreateFileMapping(INVALID_HANDLE_VALUE,
|
|
NULL, PAGE_READWRITE, 0, MAX_EUDC_CHARS * uWriteLen + 20,
|
|
lpImeL->szUsrDicMap);
|
|
|
|
if (!lpInstL->hUsrDicMem) {
|
|
MemoryLess(
|
|
#if defined(UNIIME)
|
|
lpInstL, lpImeL,
|
|
#endif
|
|
ERRMSG_MEM_USRDIC);
|
|
fRet = FALSE;
|
|
} else if (GetLastError() == ERROR_ALREADY_EXISTS) {
|
|
// another process also create another one, we will use it
|
|
hReadUsrDicMem = OpenFileMapping(FILE_MAP_READ, FALSE,
|
|
lpImeL->szUsrDicMap);
|
|
CloseHandle(lpInstL->hUsrDicMem);
|
|
CloseHandle(hUsrDicFile);
|
|
|
|
if (hReadUsrDicMem != NULL) { // OK
|
|
lpInstL->hUsrDicMem = hReadUsrDicMem;
|
|
lpImeL->uUsrDicSize = uUsrDicSize;
|
|
lpImeL->fdwErrMsg &= ~(ERRMSG_MEM_USRDIC);
|
|
} else {
|
|
MemoryLess(
|
|
#if defined(UNIIME)
|
|
lpInstL, lpImeL,
|
|
#endif
|
|
ERRMSG_MEM_USRDIC);
|
|
lpInstL->hUsrDicMem = NULL;
|
|
}
|
|
|
|
return;
|
|
} else {
|
|
fRet = ReadUsrDicToMem(
|
|
#if defined(UNIIME)
|
|
lpInstL, lpImeL,
|
|
#endif
|
|
hUsrDicFile, dwUsrDicFileSize, uUsrDicSize, uRecLen,
|
|
uReadLen, uWriteLen);
|
|
}
|
|
|
|
CloseHandle(hUsrDicFile);
|
|
|
|
if (!fRet) {
|
|
if (lpInstL->hUsrDicMem) {
|
|
CloseHandle(lpInstL->hUsrDicMem);
|
|
lpInstL->hUsrDicMem = NULL;
|
|
}
|
|
return;
|
|
}
|
|
|
|
// open a read only memory for EUDC table
|
|
hReadUsrDicMem = OpenFileMapping(FILE_MAP_READ, FALSE,
|
|
lpImeL->szUsrDicMap);
|
|
|
|
// reopen a read file and close the original write file
|
|
CloseHandle(lpInstL->hUsrDicMem);
|
|
|
|
lpImeL->uUsrDicSize = uUsrDicSize;
|
|
|
|
LoadUsrDicMem:
|
|
lpInstL->hUsrDicMem = hReadUsrDicMem;
|
|
LoadUsrDicErrMsg:
|
|
lpImeL->fdwErrMsg &= ~(ERRMSG_LOAD_USRDIC | ERRMSG_MEM_USRDIC);
|
|
|
|
return;
|
|
}
|
|
|
|
/**********************************************************************/
|
|
/* LoadOneTable() */
|
|
/* Description: */
|
|
/* memory handle & size of .TBL file will be assigned to */
|
|
/* lpImeL */
|
|
/* Eeturn Value: */
|
|
/* length of directory of the .TBL file */
|
|
/**********************************************************************/
|
|
UINT PASCAL LoadOneTable( // load one of table file
|
|
#if defined(UNIIME)
|
|
LPINSTDATAL lpInstL,
|
|
LPIMEL lpImeL,
|
|
#endif
|
|
LPTSTR szTable, // file name of .TBL
|
|
UINT uIndex, // the index of array to store memory handle
|
|
UINT uLen, // length of the directory
|
|
LPTSTR szPath) // buffer for directory
|
|
{
|
|
HANDLE hTblFile;
|
|
HGLOBAL hMap;
|
|
DWORD dwFileSize;
|
|
PSECURITY_ATTRIBUTES psa;
|
|
|
|
if (lpInstL->hMapTbl[uIndex]) { // already loaded
|
|
CloseHandle(lpInstL->hMapTbl[uIndex]);
|
|
lpInstL->hMapTbl[uIndex] = (HANDLE)NULL;
|
|
}
|
|
|
|
psa = CreateSecurityAttributes();
|
|
|
|
if (uLen) {
|
|
lstrcpy((LPTSTR)&szPath[uLen], szTable);
|
|
hTblFile = CreateFile(szPath, GENERIC_READ,
|
|
FILE_SHARE_READ|FILE_SHARE_WRITE,
|
|
psa, OPEN_EXISTING,
|
|
FILE_ATTRIBUTE_NORMAL, (HANDLE)NULL);
|
|
|
|
if (hTblFile != INVALID_HANDLE_VALUE) {
|
|
goto OpenDicFile;
|
|
}
|
|
|
|
// if the work station version, SHARE_WRITE will fail
|
|
hTblFile = CreateFile(szPath, GENERIC_READ,
|
|
FILE_SHARE_READ, psa,
|
|
OPEN_EXISTING, FILE_ATTRIBUTE_NORMAL, (HANDLE)NULL);
|
|
} else {
|
|
// try system directory next
|
|
uLen = GetSystemDirectory(szPath, MAX_PATH);
|
|
if (szPath[uLen - 1] != '\\') { // consider N:\ ;
|
|
szPath[uLen++] = '\\';
|
|
}
|
|
|
|
lstrcpy((LPTSTR)&szPath[uLen], szTable);
|
|
hTblFile = CreateFile(szPath, GENERIC_READ,
|
|
FILE_SHARE_READ|FILE_SHARE_WRITE,
|
|
psa, OPEN_EXISTING,
|
|
FILE_ATTRIBUTE_NORMAL, (HANDLE)NULL);
|
|
|
|
if (hTblFile != INVALID_HANDLE_VALUE) {
|
|
goto OpenDicFile;
|
|
}
|
|
|
|
// if the work station version, SHARE_WRITE will fail
|
|
hTblFile = CreateFile(szPath, GENERIC_READ,
|
|
FILE_SHARE_READ, psa,
|
|
OPEN_EXISTING, FILE_ATTRIBUTE_NORMAL, (HANDLE)NULL);
|
|
}
|
|
|
|
OpenDicFile:
|
|
// can not find the table file
|
|
if (hTblFile != INVALID_HANDLE_VALUE) { // OK
|
|
} else if (lpImeL->fdwErrMsg & (ERRMSG_LOAD_0 << uIndex)) {
|
|
// already prompt error message before, no more
|
|
FreeSecurityAttributes(psa);
|
|
return (0);
|
|
} else { // prompt error message
|
|
TCHAR szFmtStr[64];
|
|
TCHAR szErrMsg[2 * MAX_PATH];
|
|
HRESULT hr;
|
|
#if defined(WINAR30)
|
|
if(uIndex==4 || uIndex==5)
|
|
{
|
|
return (uLen);
|
|
}
|
|
#endif
|
|
|
|
// temp use szIMEName as format string buffer of error message
|
|
LoadString(hInst, IDS_FILE_OPEN_ERR, szFmtStr, sizeof(szFmtStr)/sizeof(TCHAR));
|
|
hr = StringCchPrintf(szErrMsg, ARRAYSIZE(szErrMsg), szFmtStr, szTable);
|
|
if (FAILED(hr))
|
|
return 0;
|
|
|
|
lpImeL->fdwErrMsg |= ERRMSG_LOAD_0 << uIndex;
|
|
MessageBeep((UINT)-1);
|
|
MessageBox((HWND)NULL, szErrMsg, lpImeL->szIMEName,
|
|
MB_OK|MB_ICONHAND|MB_TASKMODAL|MB_TOPMOST);
|
|
FreeSecurityAttributes(psa);
|
|
return (0);
|
|
}
|
|
|
|
lpImeL->fdwErrMsg &= ~(ERRMSG_LOAD_0 << uIndex);
|
|
|
|
// create file mapping for IME tables
|
|
hMap = CreateFileMapping((HANDLE)hTblFile, psa, PAGE_READONLY,
|
|
0, 0, szTable);
|
|
|
|
dwFileSize = GetFileSize(hTblFile, (LPDWORD)NULL);
|
|
|
|
CloseHandle(hTblFile);
|
|
|
|
FreeSecurityAttributes(psa);
|
|
|
|
if (!hMap) {
|
|
MemoryLess(
|
|
#if defined(UNIIME)
|
|
lpInstL, lpImeL,
|
|
#endif
|
|
ERRMSG_MEM_0 << uIndex);
|
|
return (0);
|
|
}
|
|
|
|
lpImeL->fdwErrMsg &= ~(ERRMSG_MEM_0 << uIndex);
|
|
|
|
lpInstL->hMapTbl[uIndex] = hMap;
|
|
|
|
// get file length
|
|
lpImeL->uTblSize[uIndex] = dwFileSize;
|
|
return (uLen);
|
|
}
|
|
#endif
|
|
|
|
/**********************************************************************/
|
|
/* LoadTable() */
|
|
/* Return Value: */
|
|
/* TRUE - successful, FALSE - failure */
|
|
/**********************************************************************/
|
|
BOOL PASCAL LoadTable( // check the table files of IME, include user
|
|
// defined dictionary
|
|
LPINSTDATAL lpInstL,
|
|
LPIMEL lpImeL)
|
|
{
|
|
#if !defined(WINIME) && !defined(UNICDIME)
|
|
int i;
|
|
UINT uLen;
|
|
TCHAR szBuf[MAX_PATH];
|
|
#endif
|
|
|
|
if (lpInstL->fdwTblLoad == TBL_LOADED) {
|
|
return (TRUE);
|
|
}
|
|
|
|
#if !defined(WINIME) && !defined(UNICDIME)
|
|
uLen = 0;
|
|
|
|
// A15.TBL, A234.TBL, ACODE.TBL, / PHON.TBL, PHONPTR.TBL, PHONCODE.TBL,
|
|
|
|
for (i = 0; i < MAX_IME_TABLES; i++) {
|
|
if (!*lpImeL->szTblFile[i]) {
|
|
} else if (uLen = LoadOneTable(
|
|
#if defined(UNIIME)
|
|
lpInstL, lpImeL,
|
|
#endif
|
|
lpImeL->szTblFile[i], i, uLen, szBuf)) {
|
|
} else {
|
|
int j;
|
|
|
|
for (j = 0; j < i; j++) {
|
|
if (lpInstL->hMapTbl[j]) {
|
|
CloseHandle(lpInstL->hMapTbl[j]);
|
|
lpInstL->hMapTbl[j] = (HANDLE)NULL;
|
|
}
|
|
}
|
|
|
|
lpInstL->fdwTblLoad = TBL_LOADERR;
|
|
return (FALSE);
|
|
}
|
|
}
|
|
#endif
|
|
|
|
lpInstL->fdwTblLoad = TBL_LOADED;
|
|
|
|
#if !defined(WINIME) && !defined(UNICDIME)
|
|
if (lpImeL->szUsrDic[0]) {
|
|
LoadUsrDicFile(lpInstL, lpImeL);
|
|
}
|
|
#endif
|
|
|
|
return (TRUE);
|
|
}
|
|
|
|
/**********************************************************************/
|
|
/* FreeTable() */
|
|
/**********************************************************************/
|
|
void PASCAL FreeTable(
|
|
LPINSTDATAL lpInstL)
|
|
{
|
|
#if !defined(WINIME) && !defined(UNICDIME)
|
|
int i;
|
|
|
|
// A15.TBL, A234.TBL, ACODE.TBL, / PHON.TBL, PHONPTR.TBL, PHONCODE.TBL,
|
|
|
|
for (i = 0; i < MAX_IME_TABLES; i++) {
|
|
if (lpInstL->hMapTbl[i]) {
|
|
CloseHandle(lpInstL->hMapTbl[i]);
|
|
lpInstL->hMapTbl[i] = (HANDLE)NULL;
|
|
}
|
|
}
|
|
|
|
// do not need to free phrase data base, maybe next IME will use it
|
|
// uniime.dll will free it on library detach time
|
|
|
|
if (lpInstL->hUsrDicMem) {
|
|
CloseHandle(lpInstL->hUsrDicMem);
|
|
lpInstL->hUsrDicMem = (HANDLE)NULL;
|
|
}
|
|
#endif
|
|
|
|
lpInstL->fdwTblLoad = TBL_NOTLOADED;
|
|
|
|
return;
|
|
}
|
|
#endif // !defined(ROMANIME)
|