2020-09-30 16:53:55 +02:00

293 lines
8.5 KiB
C++

/****************************************************************************
HANJA.CPP
Owner: cslim
Copyright (c) 1997-1999 Microsoft Corporation
Hanja conversion and dictionary lookup functions. Dictionary index is
stored as globally shared memory.
History:
26-APR-1999 cslim Modified for Multibox Applet Tooltip display
14-JUL-1999 cslim Copied from IME98 source tree
*****************************************************************************/
#include <windows.h>
#include <windowsx.h>
#include <stdio.h>
#include <stdlib.h>
#include "hwxobj.h"
#include "lexheader.h"
#include "hanja.h"
#include "common.h"
#include "immsec.h"
#include "dbg.h"
// NT5 Globally shared memory.
const TCHAR IMEKR_LEX_SHAREDDATA_MUTEX_NAME[] = TEXT("ImeKrLex.Mutex");
const TCHAR IMEKR_LEX_SHAREDDATA_NAME[] = TEXT("ImeKrLexHanjaToHangul.SharedMemory");
UINT vuNumofK0=0, vuNumofK1=0;
WCHAR vwcHangul=0;
// Private data
static BOOL vfLexOpen = FALSE;
static HANDLE vhLex=0;
static HANDLE vhLexIndexTbl=0;
static UINT vuNumOfHanjaEntry=0;
static DWORD viBufferStart=0; // seek point
// Private functions
static BOOL OpenLex();
//static VOID ClearHanjaSenseArray();
static INT SearchHanjaIndex(WCHAR wHChar, HanjaToHangulIndex *pLexIndexTbl);
BOOL EnsureHanjaLexLoaded()
{
_DictHeader *pLexHeader;
HKEY hKey;
DWORD dwReadBytes;
CHAR szLexFileName[MAX_PATH], szLexPathExpanded[MAX_PATH];
DWORD dwCb, dwType;
if (vfLexOpen)
return TRUE;
// Get Lex file path
szLexFileName[0] = 0;
szLexPathExpanded[0] = 0;
if (RegOpenKeyEx(HKEY_LOCAL_MACHINE, g_szIMERootKey, 0, KEY_READ, &hKey) == ERROR_SUCCESS)
{
dwCb = sizeof(szLexFileName);
dwType = REG_SZ;
if (RegQueryValueEx(hKey, g_szDictionary, NULL, &dwType, (LPBYTE)szLexFileName, &dwCb) == ERROR_SUCCESS)
ExpandEnvironmentStrings(szLexFileName, szLexPathExpanded, sizeof(szLexPathExpanded));
RegCloseKey(hKey);
}
DBGAssert(szLexPathExpanded[0] != 0);
if (szLexPathExpanded[0] == 0)
return FALSE;
vhLex = CreateFile(szLexPathExpanded, GENERIC_READ, FILE_SHARE_READ, 0, OPEN_EXISTING, FILE_FLAG_RANDOM_ACCESS, NULL);
if (vhLex==INVALID_HANDLE_VALUE)
{
DBGAssert(0);
return FALSE;
}
pLexHeader = new _DictHeader;
if (!pLexHeader)
return FALSE;
if (ReadFile(vhLex, pLexHeader, sizeof(_DictHeader), &dwReadBytes, 0) == 0 || (dwReadBytes != sizeof(_DictHeader)))
{
DBGAssert(0);
return FALSE;
}
// Set member vars
vuNumOfHanjaEntry = pLexHeader->uiNumofHanja;
viBufferStart = pLexHeader->iBufferStart;
if (pLexHeader->Version < LEX_VERSION || pLexHeader->Version > LEX_COMPATIBLE_VERSION_LIMIT )
{
delete pLexHeader;
DBGAssert(0);
return FALSE;
}
if (lstrcmpA(pLexHeader->COPYRIGHT_HEADER, COPYRIGHT_STR))
{
delete pLexHeader;
DBGAssert(0);
return FALSE;
}
// Read Index table
SetFilePointer(vhLex, pLexHeader->iHanjaToHangulIndex, 0, FILE_BEGIN);
delete pLexHeader;
return OpenLex();
}
__inline BOOL DoEnterCriticalSection(HANDLE hMutex)
{
if(WAIT_FAILED==WaitForSingleObject(hMutex, 3000)) // Wait 3 seconds
return(FALSE);
return(TRUE);
}
BOOL OpenLex()
{
BOOL fRet = FALSE;
HanjaToHangulIndex* pHanjaToHangulIndex;
HANDLE hMutex;
DWORD dwReadBytes;
///////////////////////////////////////////////////////////////////////////
// Mapping Lex file
// The dictionary index is shared data between all IME instance
hMutex=CreateMutex(GetIMESecurityAttributes(), FALSE, IMEKR_LEX_SHAREDDATA_MUTEX_NAME);
if (hMutex != NULL)
{
if (DoEnterCriticalSection(hMutex) == FALSE)
goto ExitOpenLexCritSection;
vhLexIndexTbl = OpenFileMapping(FILE_MAP_READ, TRUE, IMEKR_LEX_SHAREDDATA_NAME);
if(vhLexIndexTbl)
{
Dbg(("CHanja::OpenLex() - File mapping already exists"));
fRet = TRUE;
}
else
{
// if no file mapping exist
vhLexIndexTbl = CreateFileMapping(INVALID_HANDLE_VALUE,
GetIMESecurityAttributes(),
PAGE_READWRITE,
0,
sizeof(HanjaToHangulIndex)*(vuNumOfHanjaEntry),
IMEKR_LEX_SHAREDDATA_NAME);
if (vhLexIndexTbl)
{
Dbg(("CHanja::OpenLex() - File mapping Created"));
pHanjaToHangulIndex = (HanjaToHangulIndex*)MapViewOfFile(vhLexIndexTbl, FILE_MAP_READ | FILE_MAP_WRITE, 0, 0, 0);
if (!pHanjaToHangulIndex)
goto ExitOpenLexCritSection;
if (ReadFile(vhLex, pHanjaToHangulIndex, sizeof(HanjaToHangulIndex)*(vuNumOfHanjaEntry), &dwReadBytes, 0) != 0 &&
dwReadBytes == sizeof(HanjaToHangulIndex)*(vuNumOfHanjaEntry))
{
fRet = TRUE;
}
else
{
fRet = FALSE;
}
UnmapViewOfFile(pHanjaToHangulIndex);
}
#ifdef _DEBUG
else
DBGAssert(0);
#endif
}
ExitOpenLexCritSection:
ReleaseMutex(hMutex);
CloseHandle(hMutex);
}
FreeIMESecurityAttributes();
vfLexOpen = fRet;
return fRet;
}
BOOL CloseLex()
{
//ClearHanjaSenseArray();
if (vhLexIndexTbl)
{
CloseHandle(vhLexIndexTbl);
vhLexIndexTbl = 0;
}
if (vhLex)
{
CloseHandle(vhLex);
vhLex = 0;
}
vfLexOpen = FALSE;
return TRUE;
}
BOOL GetMeaningAndProunc(WCHAR wch, LPWSTR lpwstrTip, INT cchMax)
{
HanjaToHangulIndex* pHanjaToHangulIndex;
INT iMapHanjaInfo;
WCHAR wcHanja;
BYTE cchMeaning = 0;
WCHAR wszMeaning[MAX_SENSE_LENGTH];
DWORD dwReadBytes;
BOOL fRet = FALSE;
Dbg(("GetMeaningAndProunc"));
if (!EnsureHanjaLexLoaded())
return FALSE;
pHanjaToHangulIndex = (HanjaToHangulIndex*)MapViewOfFile(vhLexIndexTbl, FILE_MAP_READ, 0, 0, 0);
if (!pHanjaToHangulIndex)
{
DBGAssert(0);
return FALSE;
}
// Search index
if ((iMapHanjaInfo = SearchHanjaIndex(wch, pHanjaToHangulIndex)) >= 0)
{
// Seek to mapping Hanja
SetFilePointer(vhLex, viBufferStart + pHanjaToHangulIndex[iMapHanjaInfo].iOffset, 0, FILE_BEGIN);
// Read Hanja Info
if (ReadFile(vhLex, &wcHanja, sizeof(WCHAR), &dwReadBytes, 0) == 0)
{
goto GetMeaningAndProuncExit;
}
DBGAssert(wch == wcHanja);
if (ReadFile(vhLex, &cchMeaning, sizeof(BYTE), &dwReadBytes, 0) == 0)
{
goto GetMeaningAndProuncExit;
}
if (cchMeaning)
{
if (ReadFile(vhLex, wszMeaning, cchMeaning, &dwReadBytes, 0) == 0)
{
goto GetMeaningAndProuncExit;
}
}
wszMeaning[cchMeaning>>1] = L'\0';
swprintf(lpwstrTip, L"%s %c\nU+%04X", wszMeaning, pHanjaToHangulIndex[iMapHanjaInfo].wchHangul, wch);
fRet = TRUE;
}
GetMeaningAndProuncExit:
UnmapViewOfFile(pHanjaToHangulIndex);
return fRet;
}
INT SearchHanjaIndex(WCHAR wHChar, HanjaToHangulIndex *pLexIndexTbl)
{
int iHead = 0, iTail = vuNumOfHanjaEntry-1, iMid;
while (iHead <= iTail)
{
iMid = (iHead + iTail) >> 1;
Dbg(("SearchHanjaIndex iMid=%d, pLexIndexTbl[iMid].wchHanja = 0x%04X", iMid, pLexIndexTbl[iMid].wchHanja));
if (pLexIndexTbl[iMid].wchHanja > wHChar)
iTail = iMid - 1;
else
if (pLexIndexTbl[iMid].wchHanja < wHChar)
iHead = iMid + 1;
else
return (iMid);
}
return (-1);
}